/* * Minimalist Object Storage, (C) 2015 Minio, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package main import ( "fmt" "os" "os/user" "path" "runtime" "strconv" "strings" "time" "errors" "reflect" "github.com/dustin/go-humanize" "github.com/minio-io/cli" "github.com/minio-io/minio/pkg/api" "github.com/minio-io/minio/pkg/api/web" "github.com/minio-io/minio/pkg/iodine" "github.com/minio-io/minio/pkg/server/httpserver" "github.com/minio-io/minio/pkg/storage/drivers/donut" "github.com/minio-io/minio/pkg/storage/drivers/memory" "github.com/minio-io/minio/pkg/utils/log" ) var globalDebugFlag = false var commands = []cli.Command{ modeCmd, } var modeCommands = []cli.Command{ memoryCmd, donutCmd, } var modeCmd = cli.Command{ Name: "mode", Subcommands: modeCommands, Description: "Mode of execution", } var memoryCmd = cli.Command{ Name: "memory", Description: "Limit maximum memory usage to SIZE in [B, KB, MB, GB]", Action: runMemory, CustomHelpTemplate: `NAME: minio mode {{.Name}} - {{.Description}} USAGE: minio mode {{.Name}} SIZE EXAMPLES: 1. Limit maximum memory usage to 64MB $ minio mode {{.Name}} 64MB 2. Limit maximum memory usage to 4GB $ minio mode {{.Name}} 4GB `, } var donutCmd = cli.Command{ Name: "donut", Description: "Specify a path to instantiate donut", Action: runDonut, CustomHelpTemplate: `NAME: minio mode {{.Name}} - {{.Description}} USAGE: minio mode {{.Name}} PATH EXAMPLES: 1. Create a donut volume under "/mnt/backup" $ minio mode {{.Name}} /mnt/backup 2. Create a temporary donut volume under "/tmp" $ minio mode {{.Name}} /tmp 3. Create a donut volume under collection of paths $ minio mode {{.Name}} /mnt/backup2014feb /mnt/backup2014feb `, } type memoryFactory struct { httpserver.Config maxMemory uint64 } func (f memoryFactory) getStartServerFunc() startServerFunc { return func() (chan<- string, <-chan error) { _, _, driver := memory.Start(f.maxMemory) ctrl, status, _ := httpserver.Start(api.HTTPHandler(f.Domain, driver), f.Config) return ctrl, status } } type webFactory struct { httpserver.Config } func (f webFactory) getStartServerFunc() startServerFunc { return func() (chan<- string, <-chan error) { ctrl, status, _ := httpserver.Start(web.HTTPHandler(), f.Config) return ctrl, status } } type donutFactory struct { httpserver.Config paths []string } func (f donutFactory) getStartServerFunc() startServerFunc { return func() (chan<- string, <-chan error) { _, _, driver := donut.Start(f.paths) ctrl, status, _ := httpserver.Start(api.HTTPHandler(f.Domain, driver), f.Config) return ctrl, status } } var flags = []cli.Flag{ cli.StringFlag{ Name: "domain,d", Value: "", Usage: "domain used for routing incoming API requests", }, cli.StringFlag{ Name: "api-address,a", Value: ":9000", Usage: "address for incoming API requests", }, cli.StringFlag{ Name: "web-address,w", Value: ":9001", Usage: "address for incoming Management UI requests", }, cli.StringFlag{ Name: "cert,c", Hide: true, Value: "", Usage: "cert.pem", }, cli.StringFlag{ Name: "key,k", Hide: true, Value: "", Usage: "key.pem", }, cli.BoolFlag{ Name: "debug", Usage: "print debug information", }, } func init() { // Check for the environment early on and gracefuly report. _, err := user.Current() if err != nil { log.Fatalf("minio: Unable to obtain user's home directory. \nError: %s\n", err) } } type startServerFunc func() (chan<- string, <-chan error) func runMemory(c *cli.Context) { if len(c.Args()) < 1 { cli.ShowCommandHelpAndExit(c, "memory", 1) // last argument is exit code } apiServerConfig := getAPIServerConfig(c) maxMemory, err := humanize.ParseBytes(c.Args().First()) if err != nil { log.Fatalf("MaxMemory not a numeric value with reason: %s", err) } memoryDriver := memoryFactory{ Config: apiServerConfig, maxMemory: maxMemory, } apiServer := memoryDriver.getStartServerFunc() webServer := getWebServerConfigFunc(c) servers := []startServerFunc{apiServer, webServer} startMinio(servers) } func runDonut(c *cli.Context) { u, err := user.Current() if err != nil { log.Fatalln(err) } if len(c.Args()) < 1 { cli.ShowCommandHelpAndExit(c, "donut", 1) // last argument is exit code } // supporting multiple paths var paths []string if strings.TrimSpace(c.Args().First()) == "" { p := path.Join(u.HomeDir, "minio-storage", "donut") paths = append(paths, p) } else { for _, arg := range c.Args() { paths = append(paths, strings.TrimSpace(arg)) } } apiServerConfig := getAPIServerConfig(c) donutDriver := donutFactory{ Config: apiServerConfig, paths: paths, } apiServer := donutDriver.getStartServerFunc() webServer := getWebServerConfigFunc(c) servers := []startServerFunc{apiServer, webServer} startMinio(servers) } func getAPIServerConfig(c *cli.Context) httpserver.Config { certFile := c.String("cert") keyFile := c.String("key") if (certFile != "" && keyFile == "") || (certFile == "" && keyFile != "") { log.Fatalln("Both certificate and key must be provided to enable https") } tls := (certFile != "" && keyFile != "") return httpserver.Config{ Domain: c.GlobalString("domain"), Address: c.GlobalString("api-address"), TLS: tls, CertFile: certFile, KeyFile: keyFile, } } func getWebServerConfigFunc(c *cli.Context) startServerFunc { config := httpserver.Config{ Domain: c.GlobalString("domain"), Address: c.GlobalString("web-address"), TLS: false, CertFile: "", KeyFile: "", } webDrivers := webFactory{ Config: config, } return webDrivers.getStartServerFunc() } func startMinio(servers []startServerFunc) { var ctrlChannels []chan<- string var errChannels []<-chan error for _, server := range servers { ctrlChannel, errChannel := server() ctrlChannels = append(ctrlChannels, ctrlChannel) errChannels = append(errChannels, errChannel) } cases := createSelectCases(errChannels) for len(cases) > 0 { chosen, value, recvOk := reflect.Select(cases) switch recvOk { case true: // Status Message Received switch true { case value.Interface() != nil: // For any error received cleanup all existing channels and fail for _, ch := range ctrlChannels { close(ch) } msg := fmt.Sprintf("%q", value.Interface()) log.Fatal(iodine.New(errors.New(msg), nil)) } case false: // Channel closed, remove from list var aliveStatusChans []<-chan error for i, ch := range errChannels { if i != chosen { aliveStatusChans = append(aliveStatusChans, ch) } } // create new select cases without defunct channel errChannels = aliveStatusChans cases = createSelectCases(errChannels) } } } func createSelectCases(channels []<-chan error) []reflect.SelectCase { cases := make([]reflect.SelectCase, len(channels)) for i, ch := range channels { cases[i] = reflect.SelectCase{ Dir: reflect.SelectRecv, Chan: reflect.ValueOf(ch), } } return cases } // Tries to get os/arch/platform specific information // Returns a map of current os/arch/platform/memstats func getSystemData() map[string]string { host, err := os.Hostname() if err != nil { host = "" } memstats := &runtime.MemStats{} runtime.ReadMemStats(memstats) mem := fmt.Sprintf("Used: %s | Allocated: %s | Used-Heap: %s | Allocated-Heap: %s", humanize.Bytes(memstats.Alloc), humanize.Bytes(memstats.TotalAlloc), humanize.Bytes(memstats.HeapAlloc), humanize.Bytes(memstats.HeapSys)) platform := fmt.Sprintf("Host: %s | OS: %s | Arch: %s", host, runtime.GOOS, runtime.GOARCH) goruntime := fmt.Sprintf("Version: %s | CPUs: %s", runtime.Version(), strconv.Itoa(runtime.NumCPU())) return map[string]string{ "PLATFORM": platform, "RUNTIME": goruntime, "MEM": mem, } } func main() { // set up iodine iodine.SetGlobalState("minio.git", minioGitCommitHash) iodine.SetGlobalState("minio.starttime", time.Now().Format(time.RFC3339)) // set up app app := cli.NewApp() app.Name = "minio" app.Version = minioGitCommitHash app.Author = "Minio.io" app.Usage = "Minimalist Object Storage" app.Flags = flags app.Commands = commands app.Before = func(c *cli.Context) error { globalDebugFlag = c.GlobalBool("debug") if globalDebugFlag { app.ExtraInfo = getSystemData() } return nil } app.RunAndExitOnError() }