123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- package wdclient
- import (
- "errors"
- "fmt"
- "github.com/seaweedfs/seaweedfs/weed/pb"
- "math/rand"
- "strconv"
- "strings"
- "sync"
- "sync/atomic"
- "github.com/seaweedfs/seaweedfs/weed/glog"
- )
- const (
- maxCursorIndex = 4096
- )
- type HasLookupFileIdFunction interface {
- GetLookupFileIdFunction() LookupFileIdFunctionType
- }
- type LookupFileIdFunctionType func(fileId string) (targetUrls []string, err error)
- type Location struct {
- Url string `json:"url,omitempty"`
- PublicUrl string `json:"publicUrl,omitempty"`
- DataCenter string `json:"dataCenter,omitempty"`
- GrpcPort int `json:"grpcPort,omitempty"`
- }
- func (l Location) ServerAddress() pb.ServerAddress {
- return pb.NewServerAddressWithGrpcPort(l.Url, l.GrpcPort)
- }
- type vidMap struct {
- sync.RWMutex
- vid2Locations map[uint32][]Location
- ecVid2Locations map[uint32][]Location
- DataCenter string
- cursor int32
- cache *vidMap
- }
- func newVidMap(dataCenter string) *vidMap {
- return &vidMap{
- vid2Locations: make(map[uint32][]Location),
- ecVid2Locations: make(map[uint32][]Location),
- DataCenter: dataCenter,
- cursor: -1,
- }
- }
- func (vc *vidMap) getLocationIndex(length int) (int, error) {
- if length <= 0 {
- return 0, fmt.Errorf("invalid length: %d", length)
- }
- if atomic.LoadInt32(&vc.cursor) == maxCursorIndex {
- atomic.CompareAndSwapInt32(&vc.cursor, maxCursorIndex, -1)
- }
- return int(atomic.AddInt32(&vc.cursor, 1)) % length, nil
- }
- func (vc *vidMap) isSameDataCenter(loc *Location) bool {
- if vc.DataCenter == "" || loc.DataCenter == "" || vc.DataCenter != loc.DataCenter {
- return false
- }
- return true
- }
- func (vc *vidMap) LookupVolumeServerUrl(vid string) (serverUrls []string, err error) {
- id, err := strconv.Atoi(vid)
- if err != nil {
- glog.V(1).Infof("Unknown volume id %s", vid)
- return nil, err
- }
- locations, found := vc.GetLocations(uint32(id))
- if !found {
- return nil, fmt.Errorf("volume %d not found", id)
- }
- var sameDcServers, otherDcServers []string
- for _, loc := range locations {
- if vc.isSameDataCenter(&loc) {
- sameDcServers = append(sameDcServers, loc.Url)
- } else {
- otherDcServers = append(otherDcServers, loc.Url)
- }
- }
- rand.Shuffle(len(sameDcServers), func(i, j int) {
- sameDcServers[i], sameDcServers[j] = sameDcServers[j], sameDcServers[i]
- })
- rand.Shuffle(len(otherDcServers), func(i, j int) {
- otherDcServers[i], otherDcServers[j] = otherDcServers[j], otherDcServers[i]
- })
- // Prefer same data center
- serverUrls = append(sameDcServers, otherDcServers...)
- return
- }
- func (vc *vidMap) LookupFileId(fileId string) (fullUrls []string, err error) {
- parts := strings.Split(fileId, ",")
- if len(parts) != 2 {
- return nil, errors.New("Invalid fileId " + fileId)
- }
- serverUrls, lookupError := vc.LookupVolumeServerUrl(parts[0])
- if lookupError != nil {
- return nil, lookupError
- }
- for _, serverUrl := range serverUrls {
- fullUrls = append(fullUrls, "http://"+serverUrl+"/"+fileId)
- }
- return
- }
- func (vc *vidMap) GetVidLocations(vid string) (locations []Location, err error) {
- id, err := strconv.Atoi(vid)
- if err != nil {
- glog.V(1).Infof("Unknown volume id %s", vid)
- return nil, fmt.Errorf("Unknown volume id %s", vid)
- }
- foundLocations, found := vc.GetLocations(uint32(id))
- if found {
- return foundLocations, nil
- }
- return nil, fmt.Errorf("volume id %s not found", vid)
- }
- func (vc *vidMap) GetLocations(vid uint32) (locations []Location, found bool) {
- // glog.V(4).Infof("~ lookup volume id %d: %+v ec:%+v", vid, vc.vid2Locations, vc.ecVid2Locations)
- locations, found = vc.getLocations(vid)
- if found && len(locations) > 0 {
- return locations, found
- }
- if vc.cache != nil {
- return vc.cache.GetLocations(vid)
- }
- return nil, false
- }
- func (vc *vidMap) getLocations(vid uint32) (locations []Location, found bool) {
- vc.RLock()
- defer vc.RUnlock()
- locations, found = vc.vid2Locations[vid]
- if found && len(locations) > 0 {
- return
- }
- locations, found = vc.ecVid2Locations[vid]
- return
- }
- func (vc *vidMap) addLocation(vid uint32, location Location) {
- vc.Lock()
- defer vc.Unlock()
- glog.V(4).Infof("+ volume id %d: %+v", vid, location)
- locations, found := vc.vid2Locations[vid]
- if !found {
- vc.vid2Locations[vid] = []Location{location}
- return
- }
- for _, loc := range locations {
- if loc.Url == location.Url {
- return
- }
- }
- vc.vid2Locations[vid] = append(locations, location)
- }
- func (vc *vidMap) addEcLocation(vid uint32, location Location) {
- vc.Lock()
- defer vc.Unlock()
- glog.V(4).Infof("+ ec volume id %d: %+v", vid, location)
- locations, found := vc.ecVid2Locations[vid]
- if !found {
- vc.ecVid2Locations[vid] = []Location{location}
- return
- }
- for _, loc := range locations {
- if loc.Url == location.Url {
- return
- }
- }
- vc.ecVid2Locations[vid] = append(locations, location)
- }
- func (vc *vidMap) deleteLocation(vid uint32, location Location) {
- if vc.cache != nil {
- vc.cache.deleteLocation(vid, location)
- }
- vc.Lock()
- defer vc.Unlock()
- glog.V(4).Infof("- volume id %d: %+v", vid, location)
- locations, found := vc.vid2Locations[vid]
- if !found {
- return
- }
- for i, loc := range locations {
- if loc.Url == location.Url {
- vc.vid2Locations[vid] = append(locations[0:i], locations[i+1:]...)
- break
- }
- }
- }
- func (vc *vidMap) deleteEcLocation(vid uint32, location Location) {
- if vc.cache != nil {
- vc.cache.deleteLocation(vid, location)
- }
- vc.Lock()
- defer vc.Unlock()
- glog.V(4).Infof("- ec volume id %d: %+v", vid, location)
- locations, found := vc.ecVid2Locations[vid]
- if !found {
- return
- }
- for i, loc := range locations {
- if loc.Url == location.Url {
- vc.ecVid2Locations[vid] = append(locations[0:i], locations[i+1:]...)
- break
- }
- }
- }
|