minio/main.go

359 lines
8.8 KiB
Go
Raw Normal View History

2015-02-08 02:54:21 -08:00
/*
* Minimalist Object Storage, (C) 2015 Minio, Inc.
2015-02-08 02:54:21 -08:00
*
* 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 (
2015-04-02 12:48:22 -07:00
"fmt"
2015-01-28 19:52:30 -08:00
"os"
2015-04-02 12:48:22 -07:00
"os/user"
"path"
2015-04-02 12:48:22 -07:00
"runtime"
"strconv"
"strings"
2015-03-25 11:49:04 -07:00
"time"
2015-01-28 16:07:53 -08:00
2015-04-22 21:31:29 -07:00
"errors"
"reflect"
"github.com/dustin/go-humanize"
"github.com/minio-io/cli"
2015-04-22 21:31:29 -07:00
"github.com/minio-io/minio/pkg/api"
"github.com/minio-io/minio/pkg/api/web"
2015-04-08 18:02:03 -07:00
"github.com/minio-io/minio/pkg/iodine"
2015-04-22 21:31:29 -07:00
"github.com/minio-io/minio/pkg/server/httpserver"
"github.com/minio-io/minio/pkg/storage/drivers/donut"
2015-04-22 21:31:29 -07:00
"github.com/minio-io/minio/pkg/storage/drivers/memory"
"github.com/minio-io/minio/pkg/utils/log"
)
2015-04-02 12:48:22 -07:00
var globalDebugFlag = false
2015-04-22 21:31:29 -07:00
var commands = []cli.Command{
modeCmd,
}
var modeCommands = []cli.Command{
memoryCmd,
donutCmd,
}
var modeCmd = cli.Command{
Name: "mode",
Subcommands: modeCommands,
Description: "Mode of execution",
2015-04-22 21:31:29 -07:00
}
var memoryCmd = cli.Command{
Name: "memory",
Description: "Limit maximum memory usage to SIZE in [B, KB, MB, GB]",
Action: runMemory,
CustomHelpTemplate: `NAME:
2015-04-23 03:34:16 -07:00
minio mode {{.Name}} - {{.Description}}
USAGE:
2015-04-23 03:34:16 -07:00
minio mode {{.Name}} SIZE
EXAMPLES:
1. Limit maximum memory usage to 64MB
2015-04-23 03:34:16 -07:00
$ minio mode {{.Name}} 64MB
2. Limit maximum memory usage to 4GB
2015-04-23 03:34:16 -07:00
$ minio mode {{.Name}} 4GB
`,
2015-04-22 21:31:29 -07:00
}
var donutCmd = cli.Command{
Name: "donut",
Description: "Specify a path to instantiate donut",
Action: runDonut,
CustomHelpTemplate: `NAME:
2015-04-23 03:34:16 -07:00
minio mode {{.Name}} - {{.Description}}
USAGE:
2015-04-23 03:34:16 -07:00
minio mode {{.Name}} PATH
EXAMPLES:
2015-04-23 03:34:16 -07:00
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
`,
2015-04-22 21:31:29 -07:00
}
type memoryFactory struct {
2015-04-23 12:43:27 -07:00
httpserver.Config
maxMemory uint64
2015-04-22 21:31:29 -07:00
}
func (f memoryFactory) getStartServerFunc() startServerFunc {
return func() (chan<- string, <-chan error) {
_, _, driver := memory.Start(f.maxMemory)
2015-04-23 12:43:27 -07:00
ctrl, status, _ := httpserver.Start(api.HTTPHandler(f.Domain, driver), f.Config)
2015-04-22 21:31:29 -07:00
return ctrl, status
}
}
type webFactory struct {
2015-04-23 12:43:27 -07:00
httpserver.Config
2015-04-22 21:31:29 -07:00
}
func (f webFactory) getStartServerFunc() startServerFunc {
return func() (chan<- string, <-chan error) {
2015-04-23 12:43:27 -07:00
ctrl, status, _ := httpserver.Start(web.HTTPHandler(), f.Config)
2015-04-22 21:31:29 -07:00
return ctrl, status
}
}
type donutFactory struct {
2015-04-23 12:43:27 -07:00
httpserver.Config
paths []string
2015-04-22 21:31:29 -07:00
}
func (f donutFactory) getStartServerFunc() startServerFunc {
return func() (chan<- string, <-chan error) {
_, _, driver := donut.Start(f.paths)
2015-04-23 12:43:27 -07:00
ctrl, status, _ := httpserver.Start(api.HTTPHandler(f.Domain, driver), f.Config)
return ctrl, status
2015-04-22 21:31:29 -07:00
}
}
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",
},
2015-04-02 12:48:22 -07:00
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)
}
}
2015-04-22 21:31:29 -07:00
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
}
2015-04-22 21:31:29 -07:00
apiServerConfig := getAPIServerConfig(c)
maxMemory, err := humanize.ParseBytes(c.Args().First())
2015-04-22 21:31:29 -07:00
if err != nil {
log.Fatalf("MaxMemory not a numeric value with reason: %s", err)
2015-04-22 21:31:29 -07:00
}
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))
}
}
2015-04-22 21:31:29 -07:00
apiServerConfig := getAPIServerConfig(c)
donutDriver := donutFactory{
Config: apiServerConfig,
paths: paths,
2015-04-22 21:31:29 -07:00
}
apiServer := donutDriver.getStartServerFunc()
webServer := getWebServerConfigFunc(c)
servers := []startServerFunc{apiServer, webServer}
startMinio(servers)
}
2015-04-23 12:43:27 -07:00
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 != "")
2015-04-23 12:43:27 -07:00
return httpserver.Config{
Domain: c.GlobalString("domain"),
Address: c.GlobalString("api-address"),
2015-03-05 21:09:16 -08:00
TLS: tls,
CertFile: certFile,
KeyFile: keyFile,
}
2015-04-22 21:31:29 -07:00
}
func getWebServerConfigFunc(c *cli.Context) startServerFunc {
2015-04-23 12:43:27 -07:00
config := httpserver.Config{
Domain: c.GlobalString("domain"),
Address: c.GlobalString("web-address"),
2015-03-05 21:09:16 -08:00
TLS: false,
CertFile: "",
KeyFile: "",
}
2015-04-22 21:31:29 -07:00
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
}
2015-04-02 12:48:22 -07:00
// 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))
2015-04-02 12:48:22 -07:00
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() {
2015-03-25 11:44:43 -07:00
// set up iodine
2015-04-02 12:48:22 -07:00
iodine.SetGlobalState("minio.git", minioGitCommitHash)
2015-03-25 11:44:43 -07:00
iodine.SetGlobalState("minio.starttime", time.Now().Format(time.RFC3339))
// set up app
2015-01-28 19:52:30 -08:00
app := cli.NewApp()
app.Name = "minio"
2015-04-02 12:48:22 -07:00
app.Version = minioGitCommitHash
app.Author = "Minio.io"
app.Usage = "Minimalist Object Storage"
app.Flags = flags
2015-04-22 21:31:29 -07:00
app.Commands = commands
2015-04-02 12:48:22 -07:00
app.Before = func(c *cli.Context) error {
globalDebugFlag = c.GlobalBool("debug")
if globalDebugFlag {
app.ExtraInfo = getSystemData()
}
return nil
}
app.RunAndExitOnError()
}