2016-01-13 22:46:44 +01:00
/ *
Copyright
* /
package main
import (
2016-02-01 16:08:58 +01:00
"encoding/json"
2016-01-13 22:46:44 +01:00
fmtlog "log"
"os"
"strings"
"time"
log "github.com/Sirupsen/logrus"
2016-02-24 16:43:39 +01:00
"github.com/containous/traefik/middlewares"
"github.com/containous/traefik/provider"
2016-01-13 22:46:44 +01:00
"github.com/spf13/cobra"
"github.com/spf13/viper"
2016-02-09 22:29:01 +01:00
"net/http"
2016-01-13 22:46:44 +01:00
)
var traefikCmd = & cobra . Command {
Use : "traefik" ,
Short : "traefik, a modern reverse proxy" ,
Long : ` traefik is a modern HTTP reverse proxy and load balancer made to deploy microservices with ease .
Complete documentation is available at http : //traefik.io`,
Run : func ( cmd * cobra . Command , args [ ] string ) {
run ( )
} ,
}
var versionCmd = & cobra . Command {
Use : "version" ,
Short : "Print version" ,
Long : ` Print version ` ,
Run : func ( cmd * cobra . Command , args [ ] string ) {
fmtlog . Println ( Version + " built on the " + BuildDate )
os . Exit ( 0 )
} ,
}
var arguments = struct {
GlobalConfiguration
2016-02-02 18:03:40 +01:00
web bool
file bool
docker bool
dockerTLS bool
marathon bool
consul bool
2016-02-19 17:10:48 +01:00
consulTLS bool
2016-02-02 18:03:40 +01:00
consulCatalog bool
zookeeper bool
etcd bool
2016-02-19 17:10:48 +01:00
etcdTLS bool
2016-02-02 18:03:40 +01:00
boltdb bool
2016-02-08 21:57:32 +01:00
kubernetes bool
2016-01-13 22:46:44 +01:00
} {
GlobalConfiguration {
2016-01-29 20:34:17 +01:00
EntryPoints : make ( EntryPoints ) ,
2016-01-13 22:46:44 +01:00
Docker : & provider . Docker {
TLS : & provider . DockerTLS { } ,
} ,
2016-02-19 17:10:48 +01:00
File : & provider . File { } ,
Web : & WebProvider { } ,
Marathon : & provider . Marathon { } ,
Consul : & provider . Consul {
Kv : provider . Kv {
TLS : & provider . KvTLS { } ,
} ,
} ,
2016-02-02 18:03:40 +01:00
ConsulCatalog : & provider . ConsulCatalog { } ,
Zookeeper : & provider . Zookepper { } ,
2016-02-19 17:10:48 +01:00
Etcd : & provider . Etcd {
Kv : provider . Kv {
TLS : & provider . KvTLS { } ,
} ,
} ,
2016-02-08 21:57:32 +01:00
Boltdb : & provider . BoltDb { } ,
Kubernetes : & provider . Kubernetes { } ,
2016-01-13 22:46:44 +01:00
} ,
false ,
false ,
false ,
false ,
false ,
false ,
false ,
false ,
false ,
2016-02-02 18:03:40 +01:00
false ,
2016-02-19 17:10:48 +01:00
false ,
false ,
2016-02-08 21:57:32 +01:00
false ,
2016-01-13 22:46:44 +01:00
}
func init ( ) {
traefikCmd . AddCommand ( versionCmd )
traefikCmd . PersistentFlags ( ) . StringP ( "configFile" , "c" , "" , "Configuration file to use (TOML, JSON, YAML, HCL)." )
traefikCmd . PersistentFlags ( ) . StringP ( "graceTimeOut" , "g" , "10" , "Timeout in seconds. Duration to give active requests a chance to finish during hot-reloads" )
traefikCmd . PersistentFlags ( ) . String ( "accessLogsFile" , "log/access.log" , "Access logs file" )
traefikCmd . PersistentFlags ( ) . String ( "traefikLogsFile" , "log/traefik.log" , "Traefik logs file" )
2016-01-29 20:34:17 +01:00
traefikCmd . PersistentFlags ( ) . Var ( & arguments . EntryPoints , "entryPoints" , "Entrypoints definition using format: --entryPoints='Name:http Address::8000 Redirect.EntryPoint:https' --entryPoints='Name:https Address::4442 TLS:tests/traefik.crt,tests/traefik.key'" )
traefikCmd . PersistentFlags ( ) . Var ( & arguments . DefaultEntryPoints , "defaultEntryPoints" , "Entrypoints to be used by frontends that do not specify any entrypoint" )
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . StringP ( "logLevel" , "l" , "ERROR" , "Log level" )
traefikCmd . PersistentFlags ( ) . DurationVar ( & arguments . ProvidersThrottleDuration , "providersThrottleDuration" , time . Duration ( 2 * time . Second ) , "Backends throttle duration: minimum duration between 2 events from providers before applying a new configuration. It avoids unnecessary reloads if multiples events are sent in a short amount of time." )
2016-02-09 22:29:01 +01:00
traefikCmd . PersistentFlags ( ) . Int ( "maxIdleConnsPerHost" , 0 , "If non-zero, controls the maximum idle (keep-alive) to keep per-host. If zero, DefaultMaxIdleConnsPerHost is used" )
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . web , "web" , false , "Enable Web backend" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Web . Address , "web.address" , ":8080" , "Web administration port" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Web . CertFile , "web.cerFile" , "" , "SSL certificate" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Web . KeyFile , "web.keyFile" , "" , "SSL certificate" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Web . ReadOnly , "web.readOnly" , false , "Enable read only API" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . file , "file" , false , "Enable File backend" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . File . Watch , "file.watch" , true , "Watch provider" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . File . Filename , "file.filename" , "" , "Override default configuration template. For advanced users :)" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . docker , "docker" , false , "Enable Docker backend" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Docker . Watch , "docker.watch" , true , "Watch provider" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Docker . Filename , "docker.filename" , "" , "Override default configuration template. For advanced users :)" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Docker . Endpoint , "docker.endpoint" , "unix:///var/run/docker.sock" , "Docker server endpoint. Can be a tcp or a unix socket endpoint" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Docker . Domain , "docker.domain" , "" , "Default domain used" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . dockerTLS , "docker.tls" , false , "Enable Docker TLS support" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Docker . TLS . CA , "docker.tls.ca" , "" , "TLS CA" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Docker . TLS . Cert , "docker.tls.cert" , "" , "TLS cert" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Docker . TLS . Key , "docker.tls.key" , "" , "TLS key" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Docker . TLS . InsecureSkipVerify , "docker.tls.insecureSkipVerify" , false , "TLS insecure skip verify" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . marathon , "marathon" , false , "Enable Marathon backend" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Marathon . Watch , "marathon.watch" , true , "Watch provider" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Marathon . Filename , "marathon.filename" , "" , "Override default configuration template. For advanced users :)" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Marathon . Endpoint , "marathon.endpoint" , "http://127.0.0.1:8080" , "Marathon server endpoint. You can also specify multiple endpoint for Marathon" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Marathon . Domain , "marathon.domain" , "" , "Default domain used" )
2016-03-21 12:37:02 +03:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Marathon . ExposedByDefault , "marathon.exposedByDefault" , true , "Expose Marathon apps by default" )
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . consul , "consul" , false , "Enable Consul backend" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Consul . Watch , "consul.watch" , true , "Watch provider" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Consul . Filename , "consul.filename" , "" , "Override default configuration template. For advanced users :)" )
2016-02-25 23:28:50 +00:00
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Consul . Endpoint , "consul.endpoint" , "127.0.0.1:8500" , "Comma sepparated Consul server endpoints" )
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Consul . Prefix , "consul.prefix" , "/traefik" , "Prefix used for KV store" )
2016-02-19 17:10:48 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . consulTLS , "consul.tls" , false , "Enable Consul TLS support" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Consul . TLS . CA , "consul.tls.ca" , "" , "TLS CA" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Consul . TLS . Cert , "consul.tls.cert" , "" , "TLS cert" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Consul . TLS . Key , "consul.tls.key" , "" , "TLS key" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Consul . TLS . InsecureSkipVerify , "consul.tls.insecureSkipVerify" , false , "TLS insecure skip verify" )
2016-01-13 22:46:44 +01:00
2016-02-02 18:03:40 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . consulCatalog , "consulCatalog" , false , "Enable Consul catalog backend" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . ConsulCatalog . Domain , "consulCatalog.domain" , "" , "Default domain used" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . ConsulCatalog . Endpoint , "consulCatalog.endpoint" , "127.0.0.1:8500" , "Consul server endpoint" )
2016-04-12 09:49:37 +02:00
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . ConsulCatalog . Prefix , "consulCatalog.prefix" , "traefik" , "Consul catalog tag prefix" )
2016-02-02 18:03:40 +01:00
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . zookeeper , "zookeeper" , false , "Enable Zookeeper backend" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Zookeeper . Watch , "zookeeper.watch" , true , "Watch provider" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Zookeeper . Filename , "zookeeper.filename" , "" , "Override default configuration template. For advanced users :)" )
2016-02-25 23:28:50 +00:00
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Zookeeper . Endpoint , "zookeeper.endpoint" , "127.0.0.1:2181" , "Comma sepparated Zookeeper server endpoints" )
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Zookeeper . Prefix , "zookeeper.prefix" , "/traefik" , "Prefix used for KV store" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . etcd , "etcd" , false , "Enable Etcd backend" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Etcd . Watch , "etcd.watch" , true , "Watch provider" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Etcd . Filename , "etcd.filename" , "" , "Override default configuration template. For advanced users :)" )
2016-02-25 23:28:50 +00:00
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Etcd . Endpoint , "etcd.endpoint" , "127.0.0.1:4001" , "Comma sepparated Etcd server endpoints" )
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Etcd . Prefix , "etcd.prefix" , "/traefik" , "Prefix used for KV store" )
2016-02-19 17:10:48 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . etcdTLS , "etcd.tls" , false , "Enable Etcd TLS support" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Etcd . TLS . CA , "etcd.tls.ca" , "" , "TLS CA" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Etcd . TLS . Cert , "etcd.tls.cert" , "" , "TLS cert" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Etcd . TLS . Key , "etcd.tls.key" , "" , "TLS key" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Etcd . TLS . InsecureSkipVerify , "etcd.tls.insecureSkipVerify" , false , "TLS insecure skip verify" )
2016-01-13 22:46:44 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . boltdb , "boltdb" , false , "Enable Boltdb backend" )
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . Boltdb . Watch , "boltdb.watch" , true , "Watch provider" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Boltdb . Filename , "boltdb.filename" , "" , "Override default configuration template. For advanced users :)" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Boltdb . Endpoint , "boltdb.endpoint" , "127.0.0.1:4001" , "Boltdb server endpoint" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Boltdb . Prefix , "boltdb.prefix" , "/traefik" , "Prefix used for KV store" )
2016-02-08 21:57:32 +01:00
traefikCmd . PersistentFlags ( ) . BoolVar ( & arguments . kubernetes , "kubernetes" , false , "Enable Kubernetes backend" )
traefikCmd . PersistentFlags ( ) . StringVar ( & arguments . Kubernetes . Endpoint , "kubernetes.endpoint" , "127.0.0.1:8080" , "Kubernetes server endpoint" )
2016-03-22 01:32:02 +01:00
_ = viper . BindPFlag ( "configFile" , traefikCmd . PersistentFlags ( ) . Lookup ( "configFile" ) )
_ = viper . BindPFlag ( "graceTimeOut" , traefikCmd . PersistentFlags ( ) . Lookup ( "graceTimeOut" ) )
_ = viper . BindPFlag ( "logLevel" , traefikCmd . PersistentFlags ( ) . Lookup ( "logLevel" ) )
2016-01-13 22:46:44 +01:00
// TODO: wait for this issue to be corrected: https://github.com/spf13/viper/issues/105
2016-03-22 01:32:02 +01:00
_ = viper . BindPFlag ( "providersThrottleDuration" , traefikCmd . PersistentFlags ( ) . Lookup ( "providersThrottleDuration" ) )
_ = viper . BindPFlag ( "maxIdleConnsPerHost" , traefikCmd . PersistentFlags ( ) . Lookup ( "maxIdleConnsPerHost" ) )
2016-01-13 22:46:44 +01:00
viper . SetDefault ( "providersThrottleDuration" , time . Duration ( 2 * time . Second ) )
2016-01-29 20:34:17 +01:00
viper . SetDefault ( "logLevel" , "ERROR" )
2016-02-25 18:30:13 +01:00
viper . SetDefault ( "MaxIdleConnsPerHost" , 200 )
2016-01-13 22:46:44 +01:00
}
func run ( ) {
fmtlog . SetFlags ( fmtlog . Lshortfile | fmtlog . LstdFlags )
// load global configuration
globalConfiguration := LoadConfiguration ( )
2016-02-09 22:29:01 +01:00
http . DefaultTransport . ( * http . Transport ) . MaxIdleConnsPerHost = globalConfiguration . MaxIdleConnsPerHost
2016-01-13 22:46:44 +01:00
loggerMiddleware := middlewares . NewLogger ( globalConfiguration . AccessLogsFile )
defer loggerMiddleware . Close ( )
// logging
level , err := log . ParseLevel ( strings . ToLower ( globalConfiguration . LogLevel ) )
if err != nil {
log . Fatal ( "Error getting level" , err )
}
log . SetLevel ( level )
if len ( globalConfiguration . TraefikLogsFile ) > 0 {
fi , err := os . OpenFile ( globalConfiguration . TraefikLogsFile , os . O_RDWR | os . O_CREATE | os . O_APPEND , 0666 )
2016-03-22 01:32:02 +01:00
defer func ( ) {
if err := fi . Close ( ) ; err != nil {
log . Error ( "Error closinf file" , err )
}
} ( )
2016-01-13 22:46:44 +01:00
if err != nil {
log . Fatal ( "Error opening file" , err )
} else {
log . SetOutput ( fi )
log . SetFormatter ( & log . TextFormatter { DisableColors : true , FullTimestamp : true , DisableSorting : true } )
}
} else {
log . SetFormatter ( & log . TextFormatter { FullTimestamp : true , DisableSorting : true } )
}
2016-01-29 20:34:17 +01:00
jsonConf , _ := json . Marshal ( globalConfiguration )
log . Debugf ( "Global configuration loaded %s" , string ( jsonConf ) )
2016-01-13 22:46:44 +01:00
server := NewServer ( * globalConfiguration )
server . Start ( )
defer server . Close ( )
log . Info ( "Shutting down" )
}