2021-04-18 12:41:13 -07:00
// Copyright (c) 2015-2021 MinIO, Inc.
//
// This file is part of MinIO Object Storage stack
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
2016-07-12 23:21:18 -07:00
2016-08-18 16:23:42 -07:00
package cmd
2016-07-12 23:21:18 -07:00
import (
"fmt"
2018-12-18 08:26:30 -08:00
"net"
2021-08-05 15:01:19 -07:00
"net/url"
2016-07-12 23:21:18 -07:00
"runtime"
"strings"
2016-10-05 12:48:07 -07:00
humanize "github.com/dustin/go-humanize"
2021-05-06 08:52:02 -07:00
"github.com/minio/madmin-go"
2021-06-01 14:59:40 -07:00
color "github.com/minio/minio/internal/color"
"github.com/minio/minio/internal/logger"
2021-06-14 14:54:37 -07:00
xnet "github.com/minio/pkg/net"
2016-07-12 23:21:18 -07:00
)
// generates format string depending on the string length and padding.
func getFormatStr ( strLen int , padding int ) string {
formatStr := fmt . Sprintf ( "%ds" , strLen + padding )
return "%" + formatStr
}
2020-05-28 13:03:04 -07:00
func mustGetStorageInfo ( objAPI ObjectLayer ) StorageInfo {
2021-01-04 09:42:09 -08:00
storageInfo , _ := objAPI . StorageInfo ( GlobalContext )
2020-05-28 13:03:04 -07:00
return storageInfo
}
2016-07-12 23:21:18 -07:00
// Prints the formatted startup message.
2020-10-09 09:59:52 -07:00
func printStartupMessage ( apiEndpoints [ ] string , err error ) {
2022-06-29 16:32:04 -07:00
logger . Info ( color . Bold ( "MinIO Object Storage Server" ) )
2020-10-09 09:59:52 -07:00
if err != nil {
2022-03-03 13:21:16 -08:00
if globalConsoleSys != nil {
2022-07-05 14:45:49 -07:00
globalConsoleSys . Send ( fmt . Sprintf ( "Server startup failed with '%v', some features may be missing" , err ) )
2022-03-03 13:21:16 -08:00
}
2020-10-09 09:59:52 -07:00
}
2017-04-12 09:22:35 -07:00
2022-06-27 03:58:25 -07:00
if len ( globalSubnetConfig . APIKey ) == 0 && err == nil {
2022-06-28 16:37:40 -07:00
var builder strings . Builder
startupBanner ( & builder )
2022-06-29 16:32:04 -07:00
logger . Info ( builder . String ( ) )
2022-06-27 03:58:25 -07:00
}
2021-06-20 23:04:47 -07:00
strippedAPIEndpoints := stripStandardPorts ( apiEndpoints , globalMinioHost )
2018-03-28 14:14:06 -07:00
// If cache layer is enabled, print cache capacity.
2019-11-09 09:27:23 -08:00
cachedObjAPI := newCachedObjectLayerFn ( )
if cachedObjAPI != nil {
2020-04-09 09:30:02 -07:00
printCacheStorageInfo ( cachedObjAPI . StorageInfo ( GlobalContext ) )
2018-03-28 14:14:06 -07:00
}
2019-11-09 09:27:23 -08:00
2017-12-22 16:58:13 +05:30
// Object layer is initialized then print StorageInfo.
2019-11-09 09:27:23 -08:00
objAPI := newObjectLayerFn ( )
2017-12-22 16:58:13 +05:30
if objAPI != nil {
2020-05-28 13:03:04 -07:00
printStorageInfo ( mustGetStorageInfo ( objAPI ) )
2017-12-22 16:58:13 +05:30
}
2017-01-10 16:43:48 -08:00
// Prints credential, region and browser access.
2017-05-31 09:21:28 -07:00
printServerCommonMsg ( strippedAPIEndpoints )
2017-01-10 16:43:48 -08:00
// Prints `mc` cli configuration message chooses
// first endpoint as default.
2017-06-09 19:50:51 -07:00
printCLIAccessMsg ( strippedAPIEndpoints [ 0 ] , "myminio" )
2017-01-10 16:43:48 -08:00
// Prints documentation message.
2016-07-12 23:21:18 -07:00
printObjectAPIMsg ( )
}
2021-08-05 15:01:19 -07:00
// Returns true if input is IPv6
func isIPv6 ( host string ) bool {
2018-12-18 08:26:30 -08:00
h , _ , err := net . SplitHostPort ( host )
if err != nil {
h = host
}
ip := net . ParseIP ( h )
2021-08-05 15:01:19 -07:00
return ip . To16 ( ) != nil && ip . To4 ( ) == nil
2018-12-18 08:26:30 -08:00
}
2017-05-31 09:21:28 -07:00
// strip api endpoints list with standard ports such as
// port "80" and "443" before displaying on the startup
// banner. Returns a new list of API endpoints.
2021-06-20 23:04:47 -07:00
func stripStandardPorts ( apiEndpoints [ ] string , host string ) ( newAPIEndpoints [ ] string ) {
2021-08-05 15:01:19 -07:00
if len ( apiEndpoints ) == 1 {
return apiEndpoints
2021-07-23 15:22:25 -07:00
}
2017-05-31 09:21:28 -07:00
newAPIEndpoints = make ( [ ] string , len ( apiEndpoints ) )
// Check all API endpoints for standard ports and strip them.
for i , apiEndpoint := range apiEndpoints {
2021-08-05 15:01:19 -07:00
_ , err := xnet . ParseHTTPURL ( apiEndpoint )
if err != nil {
continue
}
u , err := url . Parse ( apiEndpoint )
2017-05-31 09:21:28 -07:00
if err != nil {
continue
}
2021-08-05 15:01:19 -07:00
if host == "" && isIPv6 ( u . Hostname ( ) ) {
// Skip all IPv6 endpoints
2018-12-18 08:26:30 -08:00
continue
}
2021-08-05 15:01:19 -07:00
if u . Port ( ) == "80" && u . Scheme == "http" || u . Port ( ) == "443" && u . Scheme == "https" {
u . Host = u . Hostname ( )
}
2018-12-18 08:26:30 -08:00
newAPIEndpoints [ i ] = u . String ( )
2017-05-31 09:21:28 -07:00
}
return newAPIEndpoints
}
2016-07-12 23:21:18 -07:00
// Prints common server startup message. Prints credential, region and browser access.
2017-01-10 16:43:48 -08:00
func printServerCommonMsg ( apiEndpoints [ ] string ) {
2016-07-12 23:21:18 -07:00
// Get saved credentials.
2019-10-22 22:59:13 -07:00
cred := globalActiveCred
2016-07-12 23:21:18 -07:00
// Get saved region.
2021-11-25 13:06:25 -08:00
region := globalSite . Region
2016-07-12 23:21:18 -07:00
2017-01-10 16:43:48 -08:00
apiEndpointStr := strings . Join ( apiEndpoints , " " )
2017-05-31 09:21:28 -07:00
2016-07-12 23:21:18 -07:00
// Colorize the message and print.
2022-03-03 13:21:16 -08:00
logger . Info ( color . Blue ( "API: " ) + color . Bold ( fmt . Sprintf ( "%s " , apiEndpointStr ) ) )
if color . IsTerminal ( ) && ( ! globalCLIContext . Anonymous && ! globalCLIContext . JSON ) {
logger . Info ( color . Blue ( "RootUser: " ) + color . Bold ( fmt . Sprintf ( "%s " , cred . AccessKey ) ) )
logger . Info ( color . Blue ( "RootPass: " ) + color . Bold ( fmt . Sprintf ( "%s " , cred . SecretKey ) ) )
2018-10-16 13:19:12 -07:00
if region != "" {
2022-03-03 13:21:16 -08:00
logger . Info ( color . Blue ( "Region: " ) + color . Bold ( fmt . Sprintf ( getFormatStr ( len ( region ) , 2 ) , region ) ) )
2018-10-16 13:19:12 -07:00
}
2017-05-15 18:17:02 -07:00
}
2016-09-14 01:11:03 -07:00
printEventNotifiers ( )
2021-04-29 19:01:43 -07:00
2022-06-27 03:58:25 -07:00
if globalMinioConsolePortAuto && globalBrowserEnabled {
logger . Info ( color . RedBold ( "\nWARNING: Console endpoint is listening on a dynamic port (%s), please use --console-address \":PORT\" to choose a static port." ,
globalMinioConsolePort ) )
}
2021-04-29 19:01:43 -07:00
if globalBrowserEnabled {
2021-06-20 23:04:47 -07:00
consoleEndpointStr := strings . Join ( stripStandardPorts ( getConsoleEndpoints ( ) , globalMinioConsoleHost ) , " " )
2022-06-27 03:58:25 -07:00
logger . Info ( color . Blue ( "Console: " ) + color . Bold ( fmt . Sprintf ( "%s " , consoleEndpointStr ) ) )
2022-03-03 13:21:16 -08:00
if color . IsTerminal ( ) && ( ! globalCLIContext . Anonymous && ! globalCLIContext . JSON ) {
logger . Info ( color . Blue ( "RootUser: " ) + color . Bold ( fmt . Sprintf ( "%s " , cred . AccessKey ) ) )
logger . Info ( color . Blue ( "RootPass: " ) + color . Bold ( fmt . Sprintf ( "%s " , cred . SecretKey ) ) )
2021-06-17 20:27:04 -07:00
}
2021-04-29 19:01:43 -07:00
}
2016-09-13 23:30:40 +05:30
}
2021-06-17 20:27:04 -07:00
// Prints startup message for Object API acces, prints link to our SDK documentation.
func printObjectAPIMsg ( ) {
2022-03-03 13:21:16 -08:00
logger . Info ( color . Blue ( "\nDocumentation: " ) + "https://docs.min.io" )
2021-06-17 20:27:04 -07:00
}
2016-09-13 23:30:40 +05:30
// Prints bucket notification configurations.
func printEventNotifiers ( ) {
2019-11-13 17:38:05 -08:00
if globalNotificationSys == nil {
return
}
2020-04-21 22:08:32 +05:30
arns := globalNotificationSys . GetARNList ( true )
2018-03-16 01:33:41 +05:30
if len ( arns ) == 0 {
2017-05-15 18:17:02 -07:00
return
2016-09-13 23:30:40 +05:30
}
2018-03-16 01:33:41 +05:30
2019-10-04 10:35:33 -07:00
arnMsg := color . Blue ( "SQS ARNs: " )
2018-03-16 01:33:41 +05:30
for _ , arn := range arns {
2021-01-06 10:38:07 -08:00
arnMsg += color . Bold ( fmt . Sprintf ( "%s " , arn ) )
2016-09-10 02:23:28 -07:00
}
2018-03-16 01:33:41 +05:30
2022-03-03 13:21:16 -08:00
logger . Info ( arnMsg )
2016-07-12 23:21:18 -07:00
}
// Prints startup message for command line access. Prints link to our documentation
// and custom platform specific message.
2017-06-09 19:50:51 -07:00
func printCLIAccessMsg ( endPoint string , alias string ) {
2016-07-12 23:21:18 -07:00
// Get saved credentials.
2019-10-22 22:59:13 -07:00
cred := globalActiveCred
2016-07-12 23:21:18 -07:00
2021-06-17 20:27:04 -07:00
const mcQuickStartGuide = "https://docs.min.io/docs/minio-client-quickstart-guide"
2016-07-12 23:21:18 -07:00
// Configure 'mc', following block prints platform specific information for minio client.
2020-01-06 08:42:47 -08:00
if color . IsTerminal ( ) && ! globalCLIContext . Anonymous {
2022-03-03 13:21:16 -08:00
logger . Info ( color . Blue ( "\nCommand-line: " ) + mcQuickStartGuide )
2018-10-16 13:19:12 -07:00
if runtime . GOOS == globalWindowsOSName {
2020-08-17 17:39:55 -07:00
mcMessage := fmt . Sprintf ( "$ mc.exe alias set %s %s %s %s" , alias ,
2019-10-04 10:35:33 -07:00
endPoint , cred . AccessKey , cred . SecretKey )
2022-03-03 13:21:16 -08:00
logger . Info ( fmt . Sprintf ( getFormatStr ( len ( mcMessage ) , 3 ) , mcMessage ) )
2018-10-16 13:19:12 -07:00
} else {
2020-08-17 17:39:55 -07:00
mcMessage := fmt . Sprintf ( "$ mc alias set %s %s %s %s" , alias ,
2019-10-04 10:35:33 -07:00
endPoint , cred . AccessKey , cred . SecretKey )
2022-03-03 13:21:16 -08:00
logger . Info ( fmt . Sprintf ( getFormatStr ( len ( mcMessage ) , 3 ) , mcMessage ) )
2018-10-16 13:19:12 -07:00
}
2016-07-12 23:21:18 -07:00
}
}
2016-10-05 12:48:07 -07:00
// Get formatted disk/storage info message.
2016-10-09 23:03:10 -07:00
func getStorageInfoMsg ( storageInfo StorageInfo ) string {
2018-05-23 17:30:25 -07:00
var msg string
2020-03-17 15:25:00 -07:00
var mcMessage string
2020-12-21 18:35:19 +01:00
onlineDisks , offlineDisks := getOnlineOfflineDisksStats ( storageInfo . Disks )
2021-03-04 14:36:23 -08:00
if storageInfo . Backend . Type == madmin . Erasure {
2020-12-21 18:35:19 +01:00
if offlineDisks . Sum ( ) > 0 {
2020-04-12 18:08:27 -07:00
mcMessage = "Use `mc admin info` to look for latest server/disk info\n"
2020-03-17 15:25:00 -07:00
}
2020-03-21 10:02:20 -07:00
2020-12-21 18:35:19 +01:00
diskInfo := fmt . Sprintf ( " %d Online, %d Offline. " , onlineDisks . Sum ( ) , offlineDisks . Sum ( ) )
2019-10-04 10:35:33 -07:00
msg += color . Blue ( "Status:" ) + fmt . Sprintf ( getFormatStr ( len ( diskInfo ) , 8 ) , diskInfo )
2020-03-21 10:02:20 -07:00
if len ( mcMessage ) > 0 {
msg = fmt . Sprintf ( "%s %s" , mcMessage , msg )
}
2016-10-05 12:48:07 -07:00
}
return msg
}
// Prints startup message of storage capacity and erasure information.
2016-10-09 23:03:10 -07:00
func printStorageInfo ( storageInfo StorageInfo ) {
2018-05-23 17:30:25 -07:00
if msg := getStorageInfoMsg ( storageInfo ) ; msg != "" {
2022-03-03 13:21:16 -08:00
logger . Info ( msg )
2018-05-23 17:30:25 -07:00
}
2016-10-05 12:48:07 -07:00
}
2016-10-14 13:48:08 +02:00
2018-05-30 11:30:14 -07:00
func printCacheStorageInfo ( storageInfo CacheStorageInfo ) {
2019-10-04 10:35:33 -07:00
msg := fmt . Sprintf ( "%s %s Free, %s Total" , color . Blue ( "Cache Capacity:" ) ,
2020-08-24 12:11:20 -07:00
humanize . IBytes ( storageInfo . Free ) ,
humanize . IBytes ( storageInfo . Total ) )
2022-03-03 13:21:16 -08:00
logger . Info ( msg )
2018-03-28 14:14:06 -07:00
}