seaweedfs/weed/server/filer_server.go

193 lines
5.4 KiB
Go
Raw Normal View History

2014-03-31 02:28:04 +08:00
package weed_server
import (
2016-08-31 11:32:30 +08:00
"encoding/json"
"math/rand"
2014-03-31 02:28:04 +08:00
"net/http"
2016-08-31 11:32:30 +08:00
"os"
2014-03-31 11:57:25 +08:00
"strconv"
"sync"
"time"
"github.com/chrislusf/seaweedfs/weed/filer"
"github.com/chrislusf/seaweedfs/weed/filer/cassandra_store"
"github.com/chrislusf/seaweedfs/weed/filer/embedded_filer"
"github.com/chrislusf/seaweedfs/weed/filer/flat_namespace"
2016-08-31 11:32:30 +08:00
"github.com/chrislusf/seaweedfs/weed/filer/mysql_store"
2016-12-08 08:24:40 +08:00
"github.com/chrislusf/seaweedfs/weed/filer/postgres_store"
"github.com/chrislusf/seaweedfs/weed/filer/redis_store"
"github.com/chrislusf/seaweedfs/weed/glog"
"github.com/chrislusf/seaweedfs/weed/security"
"github.com/chrislusf/seaweedfs/weed/storage"
"github.com/chrislusf/seaweedfs/weed/util"
2014-03-31 02:28:04 +08:00
)
2016-08-31 11:32:30 +08:00
type filerConf struct {
MysqlConf []mysql_store.MySqlConf `json:"mysql"`
2016-09-05 14:10:22 +08:00
mysql_store.ShardingConf
PostgresConf *postgres_store.PostgresConf `json:"postgres"`
2016-08-31 11:32:30 +08:00
}
func parseConfFile(confPath string) (*filerConf, error) {
var setting filerConf
configFile, err := os.Open(confPath)
defer configFile.Close()
if err != nil {
return nil, err
}
jsonParser := json.NewDecoder(configFile)
if err = jsonParser.Decode(&setting); err != nil {
return nil, err
}
return &setting, nil
}
2014-03-31 02:28:04 +08:00
type FilerServer struct {
port string
master string
mnLock sync.RWMutex
collection string
defaultReplication string
redirectOnRead bool
2015-04-14 14:38:46 +08:00
disableDirListing bool
2015-02-08 07:35:28 +08:00
secret security.Secret
filer filer.Filer
2016-08-31 11:32:30 +08:00
maxMB int
masterNodes *storage.MasterNodes
2014-03-31 02:28:04 +08:00
}
2017-05-28 11:14:22 +08:00
func NewFilerServer(defaultMux, readonlyMux *http.ServeMux, ip string, port int, master string, dir string, collection string,
2015-04-14 14:38:46 +08:00
replication string, redirectOnRead bool, disableDirListing bool,
2016-08-31 11:32:30 +08:00
confFile string,
maxMB int,
2015-02-08 07:35:28 +08:00
secret string,
cassandra_server string, cassandra_keyspace string,
redis_server string, redis_password string, redis_database int,
) (fs *FilerServer, err error) {
2014-03-31 02:28:04 +08:00
fs = &FilerServer{
master: master,
collection: collection,
defaultReplication: replication,
redirectOnRead: redirectOnRead,
2015-04-14 14:38:46 +08:00
disableDirListing: disableDirListing,
2016-08-31 11:32:30 +08:00
maxMB: maxMB,
port: ip + ":" + strconv.Itoa(port),
2014-03-31 02:28:04 +08:00
}
2016-08-31 11:32:30 +08:00
var setting *filerConf
if confFile != "" {
setting, err = parseConfFile(confFile)
if err != nil {
return nil, err
}
} else {
setting = new(filerConf)
}
if setting.MysqlConf != nil && len(setting.MysqlConf) != 0 {
2016-09-08 11:35:54 +08:00
mysql_store := mysql_store.NewMysqlStore(setting.MysqlConf, setting.IsSharding, setting.ShardCount)
2016-08-31 11:32:30 +08:00
fs.filer = flat_namespace.NewFlatNamespaceFiler(master, mysql_store)
} else if setting.PostgresConf != nil {
fs.filer = postgres_store.NewPostgresStore(master, *setting.PostgresConf)
2016-08-31 11:32:30 +08:00
} else if cassandra_server != "" {
cassandra_store, err := cassandra_store.NewCassandraStore(cassandra_keyspace, cassandra_server)
if err != nil {
glog.Fatalf("Can not connect to cassandra server %s with keyspace %s: %v", cassandra_server, cassandra_keyspace, err)
}
fs.filer = flat_namespace.NewFlatNamespaceFiler(master, cassandra_store)
} else if redis_server != "" {
redis_store := redis_store.NewRedisStore(redis_server, redis_password, redis_database)
fs.filer = flat_namespace.NewFlatNamespaceFiler(master, redis_store)
} else {
if fs.filer, err = embedded_filer.NewFilerEmbedded(master, dir); err != nil {
2015-01-13 13:20:11 +08:00
glog.Fatalf("Can not start filer in dir %s : %v", dir, err)
return
}
2017-05-28 09:11:18 +08:00
defaultMux.HandleFunc("/admin/mv", fs.moveHandler)
defaultMux.HandleFunc("/__api__", fs.apiHandler)
2014-03-31 02:28:04 +08:00
}
defaultMux.HandleFunc("/admin/register", fs.registerHandler)
2017-05-28 09:11:18 +08:00
defaultMux.HandleFunc("/", fs.filerHandler)
2017-05-28 11:14:22 +08:00
if defaultMux != readonlyMux {
readonlyMux.HandleFunc("/", fs.readonlyFilerHandler)
}
2014-03-31 02:28:04 +08:00
go func() {
connected := true
fs.masterNodes = storage.NewMasterNodes(fs.master)
glog.V(0).Infof("Filer server bootstraps with master %s", fs.getMasterNode())
for {
glog.V(4).Infof("Filer server sending to master %s", fs.getMasterNode())
master, err := fs.detectHealthyMaster(fs.getMasterNode())
if err == nil {
if !connected {
connected = true
if fs.getMasterNode() != master {
fs.setMasterNode(master)
}
glog.V(0).Infoln("Filer Server Connected with master at", master)
}
} else {
glog.V(1).Infof("Filer Server Failed to talk with master %s: %v", fs.getMasterNode(), err)
if connected {
connected = false
}
}
if connected {
time.Sleep(time.Duration(float32(10*1e3)*(1+rand.Float32())) * time.Millisecond)
} else {
time.Sleep(time.Duration(float32(10*1e3)*0.25) * time.Millisecond)
}
}
}()
2014-03-31 02:28:04 +08:00
return fs, nil
}
2015-02-08 07:35:28 +08:00
func (fs *FilerServer) jwt(fileId string) security.EncodedJwt {
return security.GenJwt(fs.secret, fileId)
}
func (fs *FilerServer) getMasterNode() string {
fs.mnLock.RLock()
defer fs.mnLock.RUnlock()
return fs.master
}
func (fs *FilerServer) setMasterNode(masterNode string) {
fs.mnLock.Lock()
defer fs.mnLock.Unlock()
fs.master = masterNode
}
func (fs *FilerServer) detectHealthyMaster(masterNode string) (master string, e error) {
2016-06-07 14:50:27 +08:00
if e = checkMaster(masterNode); e != nil {
fs.masterNodes.Reset()
for i := 0; i <= 3; i++ {
master, e = fs.masterNodes.FindMaster()
if e != nil {
continue
} else {
if e = checkMaster(master); e == nil {
break
}
}
}
} else {
master = masterNode
}
return
}
2016-06-07 14:50:27 +08:00
func checkMaster(masterNode string) error {
statUrl := "http://" + masterNode + "/stats"
glog.V(4).Infof("Connecting to %s ...", statUrl)
_, e := util.Get(statUrl)
return e
}