2016-06-02 18:09:14 -07:00
package command
2013-12-03 23:22:26 -08:00
import (
"net/http"
"os"
"runtime"
2014-05-07 10:17:06 -07:00
"runtime/pprof"
2013-12-03 23:22:26 -08:00
"strconv"
"strings"
2014-03-15 23:03:49 -07:00
"sync"
2013-12-03 23:22:26 -08:00
"time"
2014-10-26 11:34:55 -07:00
2016-06-02 18:09:14 -07:00
"github.com/chrislusf/seaweedfs/weed/glog"
2018-05-09 23:11:54 -07:00
"github.com/chrislusf/seaweedfs/weed/pb/master_pb"
2016-06-02 18:09:14 -07:00
"github.com/chrislusf/seaweedfs/weed/server"
2018-10-11 00:05:54 -07:00
"github.com/chrislusf/seaweedfs/weed/util"
2014-10-26 11:34:55 -07:00
"github.com/gorilla/mux"
2017-01-10 01:01:12 -08:00
"github.com/soheilhy/cmux"
"google.golang.org/grpc/reflection"
2013-12-03 23:22:26 -08:00
)
2014-05-07 10:17:06 -07:00
type ServerOptions struct {
cpuprofile * string
2018-10-11 00:04:31 -07:00
v VolumeServerOptions
2014-05-07 10:17:06 -07:00
}
2014-03-30 20:57:25 -07:00
var (
2014-05-07 10:17:06 -07:00
serverOptions ServerOptions
2014-05-13 00:03:10 -07:00
filerOptions FilerOptions
2014-03-30 20:57:25 -07:00
)
2013-12-03 23:22:26 -08:00
func init ( ) {
cmdServer . Run = runServer // break init cycle
}
var cmdServer = & Command {
2014-05-26 17:34:54 -07:00
UsageLine : "server -port=8080 -dir=/tmp -volume.max=5 -ip=server_name" ,
2013-12-03 23:22:26 -08:00
Short : "start a server, including volume server, and automatically elect a master server" ,
2014-11-28 16:34:03 -08:00
Long : ` start both a volume server to provide storage spaces
2013-12-03 23:22:26 -08:00
and a master server to provide volume = > location mapping service and sequence number of file ids
2014-11-28 16:34:03 -08:00
2013-12-03 23:22:26 -08:00
This is provided as a convenient way to start both volume server and master server .
2014-04-25 22:09:42 -07:00
The servers are exactly the same as starting them separately .
2013-12-03 23:22:26 -08:00
So other volume servers can use this embedded master server also .
2014-11-28 16:34:03 -08:00
2014-04-25 22:09:42 -07:00
Optionally , one filer server can be started . Logically , filer servers should not be in a cluster .
They run with meta data on disk , not shared . So each filer server is different .
2014-11-28 16:34:03 -08:00
2013-12-03 23:22:26 -08:00
` ,
}
var (
2015-02-11 21:04:43 -08:00
serverIp = cmdServer . Flag . String ( "ip" , "localhost" , "ip or server name" )
2014-09-20 23:34:13 -07:00
serverBindIp = cmdServer . Flag . String ( "ip.bind" , "0.0.0.0" , "ip address to bind to" )
2014-03-02 22:16:54 -08:00
serverMaxCpu = cmdServer . Flag . Int ( "maxCpu" , 0 , "maximum number of CPUs. 0 means all available CPUs" )
2017-01-10 01:01:12 -08:00
serverTimeout = cmdServer . Flag . Int ( "idleTimeout" , 30 , "connection idle seconds" )
2014-03-02 22:16:54 -08:00
serverDataCenter = cmdServer . Flag . String ( "dataCenter" , "" , "current volume server's data center name" )
serverRack = cmdServer . Flag . String ( "rack" , "" , "current volume server's rack name" )
2016-08-05 15:45:48 -06:00
serverWhiteListOption = cmdServer . Flag . String ( "whiteList" , "" , "comma separated Ip addresses having write permission. No limit if empty." )
2018-08-12 14:25:31 -07:00
serverPeers = cmdServer . Flag . String ( "master.peers" , "" , "all master nodes in comma separated ip:masterPort list" )
2015-02-07 15:35:28 -08:00
serverSecureKey = cmdServer . Flag . String ( "secure.secret" , "" , "secret to encrypt Json Web Token(JWT)" )
2018-10-14 23:12:43 -07:00
serverGarbageThreshold = cmdServer . Flag . Float64 ( "garbageThreshold" , 0.3 , "threshold to vacuum and reclaim spaces" )
2014-04-25 22:09:42 -07:00
masterPort = cmdServer . Flag . Int ( "master.port" , 9333 , "master server http listen port" )
masterMetaFolder = cmdServer . Flag . String ( "master.dir" , "" , "data directory to store meta data, default to same as -dir specified" )
masterVolumeSizeLimitMB = cmdServer . Flag . Uint ( "master.volumeSizeLimitMB" , 30 * 1000 , "Master stops directing writes to oversized volumes." )
2017-01-08 11:01:46 -08:00
masterVolumePreallocate = cmdServer . Flag . Bool ( "master.volumePreallocate" , false , "Preallocate disk space for volumes." )
2014-04-25 22:09:42 -07:00
masterDefaultReplicaPlacement = cmdServer . Flag . String ( "master.defaultReplicaPlacement" , "000" , "Default replication type if not specified." )
2014-03-02 22:16:54 -08:00
volumeDataFolders = cmdServer . Flag . String ( "dir" , os . TempDir ( ) , "directories to store data files. dir[,dir]..." )
2014-04-25 22:09:42 -07:00
volumeMaxDataVolumeCounts = cmdServer . Flag . String ( "volume.max" , "7" , "maximum numbers of volumes, count[,count]..." )
2018-10-11 00:04:31 -07:00
pulseSeconds = cmdServer . Flag . Int ( "pulseSeconds" , 5 , "number of seconds between heartbeats" )
2014-03-30 20:57:25 -07:00
isStartingFiler = cmdServer . Flag . Bool ( "filer" , false , "whether to start filer" )
2013-12-03 23:22:26 -08:00
2016-08-05 15:45:48 -06:00
serverWhiteList [ ] string
2013-12-03 23:22:26 -08:00
)
2014-03-30 20:57:25 -07:00
func init ( ) {
2014-11-28 16:34:03 -08:00
serverOptions . cpuprofile = cmdServer . Flag . String ( "cpuprofile" , "" , "cpu profile output file" )
2014-05-13 00:03:10 -07:00
filerOptions . collection = cmdServer . Flag . String ( "filer.collection" , "" , "all data will be stored in this collection" )
filerOptions . port = cmdServer . Flag . Int ( "filer.port" , 8888 , "filer server http listen port" )
2018-08-13 01:20:49 -07:00
filerOptions . grpcPort = cmdServer . Flag . Int ( "filer.port.grpc" , 0 , "filer grpc server listen port, default to http port + 10000" )
2017-05-27 20:14:22 -07:00
filerOptions . publicPort = cmdServer . Flag . Int ( "filer.port.public" , 0 , "filer server public http listen port" )
2014-05-13 00:03:10 -07:00
filerOptions . defaultReplicaPlacement = cmdServer . Flag . String ( "filer.defaultReplicaPlacement" , "" , "Default replication type if not specified during runtime." )
2014-12-08 20:27:26 -08:00
filerOptions . redirectOnRead = cmdServer . Flag . Bool ( "filer.redirectOnRead" , false , "whether proxy or redirect to volume server during file GET request" )
2015-04-13 23:38:46 -07:00
filerOptions . disableDirListing = cmdServer . Flag . Bool ( "filer.disableDirListing" , false , "turn off directory listing" )
2017-09-15 08:24:30 -07:00
filerOptions . maxMB = cmdServer . Flag . Int ( "filer.maxMB" , 32 , "split files larger than the limit" )
2018-07-08 02:11:36 -07:00
filerOptions . dirListingLimit = cmdServer . Flag . Int ( "filer.dirListLimit" , 1000 , "limit sub dir listing size" )
2018-10-11 00:04:31 -07:00
serverOptions . v . port = cmdServer . Flag . Int ( "volume.port" , 8080 , "volume server http listen port" )
serverOptions . v . publicPort = cmdServer . Flag . Int ( "volume.port.public" , 0 , "volume server public port" )
serverOptions . v . indexType = cmdServer . Flag . String ( "volume.index" , "memory" , "Choose [memory|leveldb|boltdb|btree] mode for memory~performance balance." )
2018-11-21 16:50:42 -08:00
serverOptions . v . fixJpgOrientation = cmdServer . Flag . Bool ( "volume.images.fix.orientation" , false , "Adjust jpg orientation when uploading." )
2018-10-11 00:04:31 -07:00
serverOptions . v . readRedirect = cmdServer . Flag . Bool ( "volume.read.redirect" , true , "Redirect moved or non-local volumes." )
serverOptions . v . publicUrl = cmdServer . Flag . String ( "volume.publicUrl" , "" , "publicly accessible address" )
2014-03-30 20:57:25 -07:00
}
2013-12-03 23:22:26 -08:00
func runServer ( cmd * Command , args [ ] string ) bool {
2015-02-07 15:35:28 -08:00
filerOptions . secretKey = serverSecureKey
2014-05-07 10:17:06 -07:00
if * serverOptions . cpuprofile != "" {
f , err := os . Create ( * serverOptions . cpuprofile )
if err != nil {
glog . Fatal ( err )
}
pprof . StartCPUProfile ( f )
defer pprof . StopCPUProfile ( )
}
2014-03-30 20:57:25 -07:00
2014-12-14 00:33:16 -08:00
if * filerOptions . redirectOnRead {
* isStartingFiler = true
}
2018-06-01 00:39:39 -07:00
master := * serverIp + ":" + strconv . Itoa ( * masterPort )
2017-05-27 20:14:22 -07:00
filerOptions . ip = serverIp
2018-10-11 00:04:31 -07:00
serverOptions . v . ip = serverIp
serverOptions . v . bindIp = serverBindIp
serverOptions . v . masters = & master
serverOptions . v . idleConnectionTimeout = serverTimeout
serverOptions . v . maxCpu = serverMaxCpu
serverOptions . v . dataCenter = serverDataCenter
serverOptions . v . rack = serverRack
serverOptions . v . pulseSeconds = pulseSeconds
2014-03-30 20:57:25 -07:00
2018-07-09 02:22:48 -07:00
filerOptions . dataCenter = serverDataCenter
2014-05-13 00:03:10 -07:00
if * filerOptions . defaultReplicaPlacement == "" {
* filerOptions . defaultReplicaPlacement = * masterDefaultReplicaPlacement
2014-03-30 20:57:25 -07:00
}
2013-12-03 23:22:26 -08:00
if * serverMaxCpu < 1 {
* serverMaxCpu = runtime . NumCPU ( )
}
runtime . GOMAXPROCS ( * serverMaxCpu )
folders := strings . Split ( * volumeDataFolders , "," )
2017-07-16 21:40:47 -07:00
if * masterVolumeSizeLimitMB > 30 * 1000 {
glog . Fatalf ( "masterVolumeSizeLimitMB should be less than 30000" )
}
2014-03-30 20:57:25 -07:00
if * masterMetaFolder == "" {
* masterMetaFolder = folders [ 0 ]
}
if err := util . TestFolderWritable ( * masterMetaFolder ) ; err != nil {
glog . Fatalf ( "Check Meta Folder (-mdir=\"%s\") Writable: %s" , * masterMetaFolder , err )
}
2018-12-05 23:24:25 -08:00
filerOptions . defaultLevelDbDirectory = masterMetaFolder
2014-03-30 20:57:25 -07:00
2016-08-05 15:45:48 -06:00
if * serverWhiteListOption != "" {
serverWhiteList = strings . Split ( * serverWhiteListOption , "," )
2013-12-03 23:22:26 -08:00
}
2014-03-30 20:57:25 -07:00
if * isStartingFiler {
go func ( ) {
2016-07-20 23:45:55 -07:00
time . Sleep ( 1 * time . Second )
2017-05-27 20:14:22 -07:00
2018-10-10 23:19:54 -07:00
filerOptions . startFiler ( )
2017-05-27 20:14:22 -07:00
2014-03-30 20:57:25 -07:00
} ( )
}
2014-03-30 11:28:04 -07:00
2014-03-15 23:03:49 -07:00
var raftWaitForMaster sync . WaitGroup
var volumeWait sync . WaitGroup
raftWaitForMaster . Add ( 1 )
volumeWait . Add ( 1 )
2013-12-03 23:22:26 -08:00
go func ( ) {
r := mux . NewRouter ( )
2014-03-25 13:46:59 -07:00
ms := weed_server . NewMasterServer ( r , * masterPort , * masterMetaFolder ,
2017-01-08 11:01:46 -08:00
* masterVolumeSizeLimitMB , * masterVolumePreallocate ,
2018-10-11 00:04:31 -07:00
* pulseSeconds , * masterDefaultReplicaPlacement , * serverGarbageThreshold ,
2016-08-05 15:45:48 -06:00
serverWhiteList , * serverSecureKey ,
2013-12-03 23:22:26 -08:00
)
2014-09-20 23:30:35 -07:00
glog . V ( 0 ) . Infoln ( "Start Seaweed Master" , util . VERSION , "at" , * serverIp + ":" + strconv . Itoa ( * masterPort ) )
2017-01-10 01:30:00 -08:00
masterListener , e := util . NewListener ( * serverBindIp + ":" + strconv . Itoa ( * masterPort ) , 0 )
2014-03-20 11:07:15 -07:00
if e != nil {
2015-01-13 17:04:41 -08:00
glog . Fatalf ( "Master startup error: %v" , e )
2013-12-03 23:22:26 -08:00
}
2013-12-09 13:27:09 -08:00
go func ( ) {
2014-03-15 23:03:49 -07:00
raftWaitForMaster . Wait ( )
2013-12-09 13:27:09 -08:00
time . Sleep ( 100 * time . Millisecond )
2018-08-12 14:25:31 -07:00
myAddress , peers := checkPeers ( * serverIp , * masterPort , * serverPeers )
2018-10-11 00:04:31 -07:00
raftServer := weed_server . NewRaftServer ( r , peers , myAddress , * masterMetaFolder , ms . Topo , * pulseSeconds )
2014-02-05 01:54:52 -08:00
ms . SetRaftServer ( raftServer )
2014-03-15 23:03:49 -07:00
volumeWait . Done ( )
2013-12-09 13:27:09 -08:00
} ( )
2014-03-15 23:03:49 -07:00
raftWaitForMaster . Done ( )
2017-01-10 01:01:12 -08:00
// start grpc and http server
m := cmux . New ( masterListener )
grpcL := m . Match ( cmux . HTTP2HeaderField ( "content-type" , "application/grpc" ) )
httpL := m . Match ( cmux . Any ( ) )
// Create your protocol servers.
2018-07-03 19:07:55 -07:00
grpcS := util . NewGrpcServer ( )
2018-05-09 23:11:54 -07:00
master_pb . RegisterSeaweedServer ( grpcS , ms )
2017-01-10 01:01:12 -08:00
reflection . Register ( grpcS )
httpS := & http . Server { Handler : r }
go grpcS . Serve ( grpcL )
go httpS . Serve ( httpL )
if err := m . Serve ( ) ; err != nil {
glog . Fatalf ( "master server failed to serve: %v" , err )
2013-12-03 23:22:26 -08:00
}
2017-01-10 01:01:12 -08:00
2013-12-03 23:22:26 -08:00
} ( )
2014-03-15 23:03:49 -07:00
volumeWait . Wait ( )
2013-12-09 13:27:09 -08:00
time . Sleep ( 100 * time . Millisecond )
2014-05-13 00:03:10 -07:00
2018-10-11 00:04:31 -07:00
serverOptions . v . startVolumeServer ( * volumeDataFolders , * volumeMaxDataVolumeCounts , * serverWhiteListOption )
2013-12-03 23:22:26 -08:00
return true
}