2016-12-18 05:21:29 +00:00
package cli
import (
2017-02-06 05:29:34 +00:00
"time"
2016-12-18 05:21:29 +00:00
"github.com/portainer/portainer"
"os"
"strings"
2016-12-25 20:34:02 +00:00
"gopkg.in/alecthomas/kingpin.v2"
2016-12-18 05:21:29 +00:00
)
// Service implements the CLIService interface
type Service struct { }
const (
2017-04-16 07:54:51 +00:00
errInvalidEndpointProtocol = portainer . Error ( "Invalid endpoint protocol: Portainer only supports unix:// or tcp://" )
errSocketNotFound = portainer . Error ( "Unable to locate Unix socket" )
errEndpointsFileNotFound = portainer . Error ( "Unable to locate external endpoints file" )
errInvalidSyncInterval = portainer . Error ( "Invalid synchronization interval" )
errEndpointExcludeExternal = portainer . Error ( "Cannot use the -H flag mutually with --external-endpoints" )
errNoAuthExcludeAdminPassword = portainer . Error ( "Cannot use --no-auth with --admin-password" )
2016-12-18 05:21:29 +00:00
)
// ParseFlags parse the CLI flags and return a portainer.Flags struct
func ( * Service ) ParseFlags ( version string ) ( * portainer . CLIFlags , error ) {
kingpin . Version ( version )
flags := & portainer . CLIFlags {
2017-02-06 05:29:34 +00:00
Endpoint : kingpin . Flag ( "host" , "Dockerd endpoint" ) . Short ( 'H' ) . String ( ) ,
Logo : kingpin . Flag ( "logo" , "URL for the logo displayed in the UI" ) . String ( ) ,
2017-03-03 11:54:22 +00:00
Labels : pairs ( kingpin . Flag ( "hide-label" , "Hide containers with a specific label in the UI" ) . Short ( 'l' ) ) ,
2017-02-06 05:29:34 +00:00
ExternalEndpoints : kingpin . Flag ( "external-endpoints" , "Path to a file defining available endpoints" ) . String ( ) ,
SyncInterval : kingpin . Flag ( "sync-interval" , "Duration between each synchronization via the external endpoints source" ) . Default ( defaultSyncInterval ) . String ( ) ,
Addr : kingpin . Flag ( "bind" , "Address and port to serve Portainer" ) . Default ( defaultBindAddress ) . Short ( 'p' ) . String ( ) ,
Assets : kingpin . Flag ( "assets" , "Path to the assets" ) . Default ( defaultAssetsDirectory ) . Short ( 'a' ) . String ( ) ,
Data : kingpin . Flag ( "data" , "Path to the folder where the data is stored" ) . Default ( defaultDataDirectory ) . Short ( 'd' ) . String ( ) ,
Templates : kingpin . Flag ( "templates" , "URL to the templates (apps) definitions" ) . Default ( defaultTemplatesURL ) . Short ( 't' ) . String ( ) ,
NoAuth : kingpin . Flag ( "no-auth" , "Disable authentication" ) . Default ( defaultNoAuth ) . Bool ( ) ,
2017-03-03 11:54:22 +00:00
NoAnalytics : kingpin . Flag ( "no-analytics" , "Disable Analytics in app" ) . Default ( defaultNoAuth ) . Bool ( ) ,
2017-02-06 05:29:34 +00:00
TLSVerify : kingpin . Flag ( "tlsverify" , "TLS support" ) . Default ( defaultTLSVerify ) . Bool ( ) ,
TLSCacert : kingpin . Flag ( "tlscacert" , "Path to the CA" ) . Default ( defaultTLSCACertPath ) . String ( ) ,
TLSCert : kingpin . Flag ( "tlscert" , "Path to the TLS certificate file" ) . Default ( defaultTLSCertPath ) . String ( ) ,
TLSKey : kingpin . Flag ( "tlskey" , "Path to the TLS key" ) . Default ( defaultTLSKeyPath ) . String ( ) ,
2017-04-16 07:54:51 +00:00
AdminPassword : kingpin . Flag ( "admin-password" , "Hashed admin password" ) . String ( ) ,
2016-12-18 05:21:29 +00:00
}
kingpin . Parse ( )
return flags , nil
}
// ValidateFlags validates the values of the flags.
func ( * Service ) ValidateFlags ( flags * portainer . CLIFlags ) error {
2017-02-07 03:26:12 +00:00
if * flags . Endpoint != "" && * flags . ExternalEndpoints != "" {
return errEndpointExcludeExternal
}
err := validateEndpoint ( * flags . Endpoint )
if err != nil {
return err
}
err = validateExternalEndpoints ( * flags . ExternalEndpoints )
if err != nil {
return err
}
err = validateSyncInterval ( * flags . SyncInterval )
if err != nil {
return err
}
2017-04-16 07:54:51 +00:00
if * flags . NoAuth && ( * flags . AdminPassword != "" ) {
return errNoAuthExcludeAdminPassword
}
2017-02-07 03:26:12 +00:00
return nil
}
func validateEndpoint ( endpoint string ) error {
if endpoint != "" {
if ! strings . HasPrefix ( endpoint , "unix://" ) && ! strings . HasPrefix ( endpoint , "tcp://" ) {
2017-04-16 07:54:51 +00:00
return errInvalidEndpointProtocol
2016-12-25 20:34:02 +00:00
}
2016-12-18 05:21:29 +00:00
2017-02-07 03:26:12 +00:00
if strings . HasPrefix ( endpoint , "unix://" ) {
socketPath := strings . TrimPrefix ( endpoint , "unix://" )
2016-12-25 20:34:02 +00:00
if _ , err := os . Stat ( socketPath ) ; err != nil {
if os . IsNotExist ( err ) {
return errSocketNotFound
}
return err
2016-12-18 05:21:29 +00:00
}
}
}
2017-02-07 03:26:12 +00:00
return nil
}
2016-12-18 05:21:29 +00:00
2017-02-07 03:26:12 +00:00
func validateExternalEndpoints ( externalEndpoints string ) error {
if externalEndpoints != "" {
if _ , err := os . Stat ( externalEndpoints ) ; err != nil {
2017-02-06 05:29:34 +00:00
if os . IsNotExist ( err ) {
return errEndpointsFileNotFound
}
return err
}
}
2017-02-07 03:26:12 +00:00
return nil
}
2017-02-06 05:29:34 +00:00
2017-02-07 03:26:12 +00:00
func validateSyncInterval ( syncInterval string ) error {
if syncInterval != defaultSyncInterval {
_ , err := time . ParseDuration ( syncInterval )
2017-02-06 05:29:34 +00:00
if err != nil {
return errInvalidSyncInterval
}
}
2016-12-18 05:21:29 +00:00
return nil
}