2017-04-15 15:49:53 +02:00
package docker
2015-09-12 15:10:03 +02:00
2015-09-08 00:15:14 +02:00
import (
2016-08-16 16:26:10 +01:00
"context"
2017-05-26 17:03:14 +02:00
"fmt"
2016-08-25 00:22:06 -04:00
"math"
2016-08-05 11:02:46 -03:00
"net"
2016-04-08 14:20:54 +02:00
"net/http"
2017-04-15 15:49:53 +02:00
"regexp"
2015-09-24 17:16:13 +02:00
"strconv"
"strings"
"text/template"
"time"
2015-09-10 22:54:37 +02:00
"github.com/BurntSushi/ty/fun"
2016-09-19 19:08:39 +02:00
"github.com/cenk/backoff"
"github.com/containous/traefik/job"
2016-09-23 18:27:01 +02:00
"github.com/containous/traefik/log"
2017-04-15 15:49:53 +02:00
"github.com/containous/traefik/provider"
2016-03-31 18:57:08 +02:00
"github.com/containous/traefik/safe"
2016-02-24 16:43:39 +01:00
"github.com/containous/traefik/types"
2016-07-21 16:33:49 +02:00
"github.com/containous/traefik/version"
2017-07-06 16:28:13 +02:00
dockertypes "github.com/docker/docker/api/types"
dockercontainertypes "github.com/docker/docker/api/types/container"
eventtypes "github.com/docker/docker/api/types/events"
"github.com/docker/docker/api/types/filters"
swarmtypes "github.com/docker/docker/api/types/swarm"
2017-10-23 10:33:02 +02:00
"github.com/docker/docker/api/types/versions"
2017-07-06 16:28:13 +02:00
"github.com/docker/docker/client"
2016-08-05 11:02:46 -03:00
"github.com/docker/go-connections/nat"
2016-04-08 14:20:54 +02:00
"github.com/docker/go-connections/sockets"
2017-10-30 15:10:05 +01:00
"github.com/pkg/errors"
2015-09-07 10:38:58 +02:00
)
2015-09-09 22:39:08 +02:00
2016-08-05 11:02:46 -03:00
const (
2017-04-15 15:49:53 +02:00
// SwarmAPIVersion is a constant holding the version of the Provider API traefik will use
2017-11-03 13:51:24 +01:00
SwarmAPIVersion = "1.24"
2016-08-05 11:02:46 -03:00
// SwarmDefaultWatchTime is the duration of the interval when polling docker
SwarmDefaultWatchTime = 15 * time . Second
2017-07-10 16:58:12 +02:00
labelDockerNetwork = "traefik.docker.network"
labelBackendLoadbalancerSwarm = "traefik.backend.loadbalancer.swarm"
labelDockerComposeProject = "com.docker.compose.project"
labelDockerComposeService = "com.docker.compose.service"
2016-08-05 11:02:46 -03:00
)
2016-04-08 14:20:54 +02:00
2017-04-15 15:49:53 +02:00
var _ provider . Provider = ( * Provider ) ( nil )
2016-08-16 19:13:18 +02:00
2017-04-17 12:50:02 +02:00
// Provider holds configurations of the provider.
2017-04-15 15:49:53 +02:00
type Provider struct {
2017-10-02 10:32:02 +02:00
provider . BaseProvider ` mapstructure:",squash" export:"true" `
2017-08-25 12:22:03 -04:00
Endpoint string ` description:"Docker server endpoint. Can be a tcp or a unix socket endpoint" `
Domain string ` description:"Default domain used" `
2017-10-02 10:32:02 +02:00
TLS * types . ClientTLS ` description:"Enable Docker TLS support" export:"true" `
ExposedByDefault bool ` description:"Expose containers by default" export:"true" `
UseBindPortIP bool ` description:"Use the ip address from the bound port, rather than from the inner network" export:"true" `
SwarmMode bool ` description:"Use Docker on Swarm Mode" export:"true" `
2016-08-05 11:02:46 -03:00
}
2017-04-15 15:49:53 +02:00
// dockerData holds the need data to the Provider p
2016-08-05 11:02:46 -03:00
type dockerData struct {
2017-02-02 20:18:12 -05:00
ServiceName string
2016-08-05 11:02:46 -03:00
Name string
Labels map [ string ] string // List of labels set to container or service
NetworkSettings networkSettings
2016-10-03 11:01:37 +02:00
Health string
2017-10-25 14:20:03 -04:00
Node * dockertypes . ContainerNode
2016-08-05 11:02:46 -03:00
}
2017-04-15 15:49:53 +02:00
// NetworkSettings holds the networks data to the Provider p
2016-08-05 11:02:46 -03:00
type networkSettings struct {
NetworkMode dockercontainertypes . NetworkMode
Ports nat . PortMap
Networks map [ string ] * networkData
}
2017-04-15 15:49:53 +02:00
// Network holds the network data to the Provider p
2016-08-05 11:02:46 -03:00
type networkData struct {
Name string
Addr string
Port int
Protocol string
ID string
2015-11-20 23:05:06 +08:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) createClient ( ) ( client . APIClient , error ) {
2016-04-08 14:20:54 +02:00
var httpClient * http . Client
httpHeaders := map [ string ] string {
2016-07-21 16:33:49 +02:00
"User-Agent" : "Traefik " + version . Version ,
2016-04-08 14:20:54 +02:00
}
2017-04-15 15:49:53 +02:00
if p . TLS != nil {
config , err := p . TLS . CreateTLSConfig ( )
2016-04-08 14:20:54 +02:00
if err != nil {
return nil , err
}
tr := & http . Transport {
TLSClientConfig : config ,
}
2017-04-15 15:49:53 +02:00
proto , addr , _ , err := client . ParseHost ( p . Endpoint )
2016-04-08 14:20:54 +02:00
if err != nil {
return nil , err
}
sockets . ConfigureTransport ( tr , proto , addr )
httpClient = & http . Client {
Transport : tr ,
}
2016-06-27 16:14:56 +02:00
2016-04-08 14:20:54 +02:00
}
2016-08-05 11:02:46 -03:00
var version string
2017-04-15 15:49:53 +02:00
if p . SwarmMode {
2016-08-05 11:02:46 -03:00
version = SwarmAPIVersion
} else {
version = DockerAPIVersion
}
2017-04-15 15:49:53 +02:00
return client . NewClient ( p . Endpoint , version , httpClient , httpHeaders )
2016-08-05 11:02:46 -03:00
2016-04-08 14:20:54 +02:00
}
2017-04-17 12:50:02 +02:00
// Provide allows the docker provider to provide configurations to traefik
2015-11-01 19:29:47 +01:00
// using the given configuration channel.
2017-04-15 15:49:53 +02:00
func ( p * Provider ) Provide ( configurationChan chan <- types . ConfigMessage , pool * safe . Pool , constraints types . Constraints ) error {
p . Constraints = append ( p . Constraints , constraints ... )
2016-04-13 20:36:23 +02:00
// TODO register this routine in pool, and watch for stop channel
2016-03-31 18:57:08 +02:00
safe . Go ( func ( ) {
2016-02-25 18:30:13 +01:00
operation := func ( ) error {
var err error
2015-11-01 19:29:47 +01:00
2017-04-15 15:49:53 +02:00
dockerClient , err := p . createClient ( )
2016-02-25 18:30:13 +01:00
if err != nil {
log . Errorf ( "Failed to create a client for docker, error: %s" , err )
return err
}
2016-06-08 19:39:38 +02:00
ctx := context . Background ( )
version , err := dockerClient . ServerVersion ( ctx )
2017-08-18 02:18:02 +02:00
if err != nil {
log . Errorf ( "Failed to retrieve information of the docker client and server host: %s" , err )
return err
}
2017-04-15 15:49:53 +02:00
log . Debugf ( "Provider connection established with docker %s (API %s)" , version . Version , version . APIVersion )
2016-08-05 11:02:46 -03:00
var dockerDataList [ ] dockerData
2017-04-15 15:49:53 +02:00
if p . SwarmMode {
dockerDataList , err = p . listServices ( ctx , dockerClient )
2016-08-05 11:02:46 -03:00
if err != nil {
log . Errorf ( "Failed to list services for docker swarm mode, error %s" , err )
return err
}
} else {
dockerDataList , err = listContainers ( ctx , dockerClient )
if err != nil {
log . Errorf ( "Failed to list containers for docker, error %s" , err )
return err
}
2016-02-25 18:30:13 +01:00
}
2016-08-05 11:02:46 -03:00
2017-04-15 15:49:53 +02:00
configuration := p . loadDockerConfig ( dockerDataList )
2016-02-25 18:30:13 +01:00
configurationChan <- types . ConfigMessage {
ProviderName : "docker" ,
Configuration : configuration ,
}
2017-04-15 15:49:53 +02:00
if p . Watch {
2016-06-08 19:39:38 +02:00
ctx , cancel := context . WithCancel ( ctx )
2017-04-15 15:49:53 +02:00
if p . SwarmMode {
2017-05-03 00:20:20 +09:00
errChan := make ( chan error )
2016-08-05 11:02:46 -03:00
// TODO: This need to be change. Linked to Swarm events docker/docker#23827
ticker := time . NewTicker ( SwarmDefaultWatchTime )
pool . Go ( func ( stop chan bool ) {
2017-05-03 00:20:20 +09:00
defer close ( errChan )
2016-08-05 11:02:46 -03:00
for {
select {
case <- ticker . C :
2017-04-15 15:49:53 +02:00
services , err := p . listServices ( ctx , dockerClient )
2016-08-05 11:02:46 -03:00
if err != nil {
log . Errorf ( "Failed to list services for docker, error %s" , err )
2017-05-03 00:20:20 +09:00
errChan <- err
2016-08-05 11:02:46 -03:00
return
}
2017-04-15 15:49:53 +02:00
configuration := p . loadDockerConfig ( services )
2016-08-05 11:02:46 -03:00
if configuration != nil {
configurationChan <- types . ConfigMessage {
ProviderName : "docker" ,
Configuration : configuration ,
}
}
case <- stop :
ticker . Stop ( )
cancel ( )
return
}
}
} )
2017-05-03 00:20:20 +09:00
if err , ok := <- errChan ; ok {
return err
}
// channel closed
2016-08-05 11:02:46 -03:00
} else {
pool . Go ( func ( stop chan bool ) {
for {
select {
case <- stop :
cancel ( )
return
}
}
} )
f := filters . NewArgs ( )
f . Add ( "type" , "container" )
options := dockertypes . EventsOptions {
Filters : f ,
}
2017-07-06 16:28:13 +02:00
2016-08-05 11:02:46 -03:00
startStopHandle := func ( m eventtypes . Message ) {
2017-04-15 15:49:53 +02:00
log . Debugf ( "Provider event received %+v" , m )
2016-08-05 11:02:46 -03:00
containers , err := listContainers ( ctx , dockerClient )
if err != nil {
log . Errorf ( "Failed to list containers for docker, error %s" , err )
// Call cancel to get out of the monitor
2016-06-16 22:49:57 +02:00
cancel ( )
return
}
2017-04-15 15:49:53 +02:00
configuration := p . loadDockerConfig ( containers )
2016-08-05 11:02:46 -03:00
if configuration != nil {
configurationChan <- types . ConfigMessage {
ProviderName : "docker" ,
Configuration : configuration ,
}
2015-09-10 22:54:37 +02:00
}
2015-09-10 09:06:37 +02:00
}
2017-07-06 16:28:13 +02:00
eventsc , errc := dockerClient . Events ( ctx , options )
for event := range eventsc {
if event . Action == "start" ||
event . Action == "die" ||
strings . HasPrefix ( event . Action , "health_status" ) {
startStopHandle ( event )
}
}
if err := <- errc ; err != nil {
2016-08-05 11:02:46 -03:00
return err
}
2016-04-08 14:20:54 +02:00
}
2015-11-01 19:29:47 +01:00
}
2016-02-25 18:30:13 +01:00
return nil
}
notify := func ( err error , time time . Duration ) {
2017-04-15 15:49:53 +02:00
log . Errorf ( "Provider connection error %+v, retrying in %s" , err , time )
2016-02-25 18:30:13 +01:00
}
2016-12-08 13:32:12 +01:00
err := backoff . RetryNotify ( safe . OperationWithRecover ( operation ) , job . NewBackOff ( backoff . NewExponentialBackOff ( ) ) , notify )
2016-02-25 18:30:13 +01:00
if err != nil {
2016-08-19 10:36:54 +02:00
log . Errorf ( "Cannot connect to docker server %+v" , err )
2016-02-25 18:30:13 +01:00
}
2016-03-31 18:57:08 +02:00
} )
2015-11-01 19:29:47 +01:00
2015-10-01 12:04:25 +02:00
return nil
2015-09-07 10:38:58 +02:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) loadDockerConfig ( containersInspected [ ] dockerData ) * types . Configuration {
2015-10-08 21:21:51 +02:00
var DockerFuncMap = template . FuncMap {
2017-04-15 15:49:53 +02:00
"getBackend" : p . getBackend ,
"getIPAddress" : p . getIPAddress ,
"getPort" : p . getPort ,
"getWeight" : p . getWeight ,
"getDomain" : p . getDomain ,
"getProtocol" : p . getProtocol ,
"getPassHostHeader" : p . getPassHostHeader ,
"getPriority" : p . getPriority ,
"getEntryPoints" : p . getEntryPoints ,
2017-04-19 11:14:05 +02:00
"getBasicAuth" : p . getBasicAuth ,
2017-04-15 15:49:53 +02:00
"getFrontendRule" : p . getFrontendRule ,
2017-11-18 13:50:03 +01:00
"getRedirect" : p . getRedirect ,
2017-04-15 15:49:53 +02:00
"hasCircuitBreakerLabel" : p . hasCircuitBreakerLabel ,
"getCircuitBreakerExpression" : p . getCircuitBreakerExpression ,
"hasLoadBalancerLabel" : p . hasLoadBalancerLabel ,
"getLoadBalancerMethod" : p . getLoadBalancerMethod ,
"hasMaxConnLabels" : p . hasMaxConnLabels ,
"getMaxConnAmount" : p . getMaxConnAmount ,
"getMaxConnExtractorFunc" : p . getMaxConnExtractorFunc ,
2017-10-16 17:38:03 +02:00
"getSticky" : p . getSticky ,
2017-10-10 11:10:02 +02:00
"getStickinessCookieName" : p . getStickinessCookieName ,
"hasStickinessLabel" : p . hasStickinessLabel ,
2017-04-15 15:49:53 +02:00
"getIsBackendLBSwarm" : p . getIsBackendLBSwarm ,
"hasServices" : p . hasServices ,
"getServiceNames" : p . getServiceNames ,
"getServicePort" : p . getServicePort ,
"getServiceWeight" : p . getServiceWeight ,
"getServiceProtocol" : p . getServiceProtocol ,
"getServiceEntryPoints" : p . getServiceEntryPoints ,
2017-04-19 11:14:05 +02:00
"getServiceBasicAuth" : p . getServiceBasicAuth ,
2017-04-15 15:49:53 +02:00
"getServiceFrontendRule" : p . getServiceFrontendRule ,
"getServicePassHostHeader" : p . getServicePassHostHeader ,
"getServicePriority" : p . getServicePriority ,
"getServiceBackend" : p . getServiceBackend ,
2017-11-18 13:50:03 +01:00
"getServiceRedirect" : p . getServiceRedirect ,
2017-04-30 11:22:07 +02:00
"getWhitelistSourceRange" : p . getWhitelistSourceRange ,
2017-10-20 10:14:03 -05:00
"getRequestHeaders" : p . getRequestHeaders ,
"getResponseHeaders" : p . getResponseHeaders ,
"hasRequestHeaders" : p . hasRequestHeaders ,
"hasResponseHeaders" : p . hasResponseHeaders ,
2015-09-10 22:54:37 +02:00
}
// filter containers
2016-08-05 11:02:46 -03:00
filteredContainers := fun . Filter ( func ( container dockerData ) bool {
2017-04-15 15:49:53 +02:00
return p . containerFilter ( container )
2016-08-05 11:02:46 -03:00
} , containersInspected ) . ( [ ] dockerData )
2015-09-10 22:54:37 +02:00
2016-08-05 11:02:46 -03:00
frontends := map [ string ] [ ] dockerData { }
2016-09-28 16:29:19 +02:00
backends := map [ string ] dockerData { }
servers := map [ string ] [ ] dockerData { }
2017-10-30 12:06:03 +01:00
for idx , container := range filteredContainers {
frontendName := p . getFrontendName ( container , idx )
2016-06-06 21:59:58 +02:00
frontends [ frontendName ] = append ( frontends [ frontendName ] , container )
2017-04-15 15:49:53 +02:00
backendName := p . getBackend ( container )
2016-09-28 16:29:19 +02:00
backends [ backendName ] = container
servers [ backendName ] = append ( servers [ backendName ] , container )
2015-09-07 10:38:58 +02:00
}
2015-09-09 16:49:51 +02:00
templateObjects := struct {
2016-09-28 13:28:20 +02:00
Containers [ ] dockerData
Frontends map [ string ] [ ] dockerData
2016-09-28 16:29:19 +02:00
Backends map [ string ] dockerData
Servers map [ string ] [ ] dockerData
2016-09-28 13:28:20 +02:00
Domain string
2015-09-07 10:38:58 +02:00
} {
2015-09-10 22:54:37 +02:00
filteredContainers ,
2015-10-23 09:49:19 +02:00
frontends ,
2016-09-28 16:29:19 +02:00
backends ,
servers ,
2017-04-15 15:49:53 +02:00
p . Domain ,
2015-09-07 10:38:58 +02:00
}
2017-04-15 15:49:53 +02:00
configuration , err := p . GetConfiguration ( "templates/docker.tmpl" , DockerFuncMap , templateObjects )
2015-09-07 10:38:58 +02:00
if err != nil {
2015-11-13 11:50:32 +01:00
log . Error ( err )
2015-09-07 10:38:58 +02:00
}
2017-11-18 13:50:03 +01:00
2015-11-13 11:50:32 +01:00
return configuration
}
2015-09-07 10:38:58 +02:00
2017-04-15 15:49:53 +02:00
func ( p * Provider ) hasCircuitBreakerLabel ( container dockerData ) bool {
2017-10-10 11:10:02 +02:00
_ , err := getLabel ( container , types . LabelBackendCircuitbreakerExpression )
return err == nil
2016-08-25 00:22:06 -04:00
}
2017-03-08 15:10:21 +01:00
// Regexp used to extract the name of the service and the name of the property for this service
// All properties are under the format traefik.<servicename>.frontent.*= except the port/weight/protocol directly after traefik.<servicename>.
2017-08-21 10:46:03 +02:00
var servicesPropertiesRegexp = regexp . MustCompile ( ` ^traefik\.(?P<service_name>.+?)\.(?P<property_name>port|weight|protocol|frontend\.(.*))$ ` )
2017-03-08 15:10:21 +01:00
// Map of services properties
// we can get it with label[serviceName][propertyName] and we got the propertyValue
type labelServiceProperties map [ string ] map [ string ] string
// Check if for the given container, we find labels that are defining services
2017-04-15 15:49:53 +02:00
func ( p * Provider ) hasServices ( container dockerData ) bool {
2017-03-08 15:10:21 +01:00
return len ( extractServicesLabels ( container . Labels ) ) > 0
}
// Extract the service labels from container labels of dockerData struct
func extractServicesLabels ( labels map [ string ] string ) labelServiceProperties {
v := make ( labelServiceProperties )
for index , serviceProperty := range labels {
matches := servicesPropertiesRegexp . FindStringSubmatch ( index )
if matches != nil {
result := make ( map [ string ] string )
for i , name := range servicesPropertiesRegexp . SubexpNames ( ) {
if i != 0 {
result [ name ] = matches [ i ]
}
}
serviceName := result [ "service_name" ]
if _ , ok := v [ serviceName ] ; ! ok {
v [ serviceName ] = make ( map [ string ] string )
}
v [ serviceName ] [ result [ "property_name" ] ] = serviceProperty
}
}
return v
}
// Gets the entry for a service label searching in all labels of the given container
func getContainerServiceLabel ( container dockerData , serviceName string , entry string ) ( string , bool ) {
value , ok := extractServicesLabels ( container . Labels ) [ serviceName ] [ entry ]
return value , ok
}
// Gets array of service names for a given container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServiceNames ( container dockerData ) [ ] string {
2017-03-08 15:10:21 +01:00
labelServiceProperties := extractServicesLabels ( container . Labels )
keys := make ( [ ] string , 0 , len ( labelServiceProperties ) )
for k := range labelServiceProperties {
keys = append ( keys , k )
}
return keys
}
// Extract entrypoints from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServiceEntryPoints ( container dockerData , serviceName string ) [ ] string {
2017-03-08 15:10:21 +01:00
if entryPoints , ok := getContainerServiceLabel ( container , serviceName , "frontend.entryPoints" ) ; ok {
return strings . Split ( entryPoints , "," )
}
2017-04-15 15:49:53 +02:00
return p . getEntryPoints ( container )
2017-03-08 15:10:21 +01:00
}
2017-04-19 11:14:05 +02:00
// Extract basic auth from labels for a given service and a given docker container
func ( p * Provider ) getServiceBasicAuth ( container dockerData , serviceName string ) [ ] string {
if basicAuth , ok := getContainerServiceLabel ( container , serviceName , "frontend.auth.basic" ) ; ok {
return strings . Split ( basicAuth , "," )
}
return p . getBasicAuth ( container )
}
2017-03-08 15:10:21 +01:00
// Extract passHostHeader from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServicePassHostHeader ( container dockerData , serviceName string ) string {
2017-03-08 15:10:21 +01:00
if servicePassHostHeader , ok := getContainerServiceLabel ( container , serviceName , "frontend.passHostHeader" ) ; ok {
return servicePassHostHeader
}
2017-04-15 15:49:53 +02:00
return p . getPassHostHeader ( container )
2017-03-08 15:10:21 +01:00
}
// Extract priority from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServicePriority ( container dockerData , serviceName string ) string {
2017-03-08 15:10:21 +01:00
if value , ok := getContainerServiceLabel ( container , serviceName , "frontend.priority" ) ; ok {
return value
}
2017-04-15 15:49:53 +02:00
return p . getPriority ( container )
2017-03-08 15:10:21 +01:00
}
// Extract backend from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServiceBackend ( container dockerData , serviceName string ) string {
2017-03-08 15:10:21 +01:00
if value , ok := getContainerServiceLabel ( container , serviceName , "frontend.backend" ) ; ok {
return value
}
2017-04-15 15:49:53 +02:00
return p . getBackend ( container ) + "-" + provider . Normalize ( serviceName )
2017-03-08 15:10:21 +01:00
}
// Extract rule from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServiceFrontendRule ( container dockerData , serviceName string ) string {
2017-03-08 15:10:21 +01:00
if value , ok := getContainerServiceLabel ( container , serviceName , "frontend.rule" ) ; ok {
return value
}
2017-04-15 15:49:53 +02:00
return p . getFrontendRule ( container )
2017-03-08 15:10:21 +01:00
}
// Extract port from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServicePort ( container dockerData , serviceName string ) string {
2017-03-08 15:10:21 +01:00
if value , ok := getContainerServiceLabel ( container , serviceName , "port" ) ; ok {
return value
}
2017-04-15 15:49:53 +02:00
return p . getPort ( container )
2017-03-08 15:10:21 +01:00
}
// Extract weight from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServiceWeight ( container dockerData , serviceName string ) string {
2017-03-08 15:10:21 +01:00
if value , ok := getContainerServiceLabel ( container , serviceName , "weight" ) ; ok {
return value
}
2017-04-15 15:49:53 +02:00
return p . getWeight ( container )
2017-03-08 15:10:21 +01:00
}
// Extract protocol from labels for a given service and a given docker container
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getServiceProtocol ( container dockerData , serviceName string ) string {
2017-03-08 15:10:21 +01:00
if value , ok := getContainerServiceLabel ( container , serviceName , "protocol" ) ; ok {
return value
}
2017-04-15 15:49:53 +02:00
return p . getProtocol ( container )
2017-03-08 15:10:21 +01:00
}
2017-11-18 13:50:03 +01:00
// Extract protocol from labels for a given service and a given docker container
func ( p * Provider ) getServiceRedirect ( container dockerData , serviceName string ) string {
if value , ok := getContainerServiceLabel ( container , serviceName , "frontend.redirect" ) ; ok {
return value
}
return p . getRedirect ( container )
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) hasLoadBalancerLabel ( container dockerData ) bool {
2017-07-10 16:58:12 +02:00
_ , errMethod := getLabel ( container , types . LabelBackendLoadbalancerMethod )
_ , errSticky := getLabel ( container , types . LabelBackendLoadbalancerSticky )
2017-10-16 17:38:03 +02:00
_ , errStickiness := getLabel ( container , types . LabelBackendLoadbalancerStickiness )
_ , errCookieName := getLabel ( container , types . LabelBackendLoadbalancerStickinessCookieName )
return errMethod == nil || errSticky == nil || errStickiness == nil || errCookieName == nil
2016-08-25 00:22:06 -04:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) hasMaxConnLabels ( container dockerData ) bool {
2017-07-10 16:58:12 +02:00
if _ , err := getLabel ( container , types . LabelBackendMaxconnAmount ) ; err != nil {
2016-08-25 00:22:06 -04:00
return false
}
2017-07-10 16:58:12 +02:00
if _ , err := getLabel ( container , types . LabelBackendMaxconnExtractorfunc ) ; err != nil {
2016-08-25 00:22:06 -04:00
return false
}
return true
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getCircuitBreakerExpression ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelBackendCircuitbreakerExpression ) ; err == nil {
2016-08-25 00:22:06 -04:00
return label
}
return "NetworkErrorRatio() > 1"
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getLoadBalancerMethod ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelBackendLoadbalancerMethod ) ; err == nil {
2016-08-25 00:22:06 -04:00
return label
}
return "wrr"
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getMaxConnAmount ( container dockerData ) int64 {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelBackendMaxconnAmount ) ; err == nil {
2016-08-25 00:22:06 -04:00
i , errConv := strconv . ParseInt ( label , 10 , 64 )
if errConv != nil {
log . Errorf ( "Unable to parse traefik.backend.maxconn.amount %s" , label )
return math . MaxInt64
}
return i
}
return math . MaxInt64
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getMaxConnExtractorFunc ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelBackendMaxconnExtractorfunc ) ; err == nil {
2016-08-25 00:22:06 -04:00
return label
}
return "request.host"
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) containerFilter ( container dockerData ) bool {
2017-11-09 12:16:03 +01:00
if ! isContainerEnabled ( container , p . ExposedByDefault ) {
log . Debugf ( "Filtering disabled container %s" , container . Name )
return false
}
2017-10-30 15:10:05 +01:00
var err error
portLabel := "traefik.port label"
if p . hasServices ( container ) {
portLabel = "traefik.<serviceName>.port or " + portLabel + "s"
err = checkServiceLabelPort ( container )
} else {
_ , err = strconv . Atoi ( container . Labels [ types . LabelPort ] )
}
2016-05-28 15:16:57 -07:00
if len ( container . NetworkSettings . Ports ) == 0 && err != nil {
2017-10-30 15:10:05 +01:00
log . Debugf ( "Filtering container without port and no %s %s : %s" , portLabel , container . Name , err . Error ( ) )
2015-11-13 11:50:32 +01:00
return false
2015-09-07 10:38:58 +02:00
}
2015-11-13 11:50:32 +01:00
2017-07-10 16:58:12 +02:00
constraintTags := strings . Split ( container . Labels [ types . LabelTags ] , "," )
2017-04-15 15:49:53 +02:00
if ok , failingConstraint := p . MatchConstraints ( constraintTags ) ; ! ok {
2016-06-06 21:59:58 +02:00
if failingConstraint != nil {
log . Debugf ( "Container %v pruned by '%v' constraint" , container . Name , failingConstraint . String ( ) )
}
return false
}
2016-10-03 11:01:37 +02:00
if container . Health != "" && container . Health != "healthy" {
log . Debugf ( "Filtering unhealthy or starting container %s" , container . Name )
return false
}
2017-04-07 16:30:57 +02:00
if len ( p . getFrontendRule ( container ) ) == 0 {
log . Debugf ( "Filtering container with empty frontend rule %s" , container . Name )
return false
}
2015-11-13 11:50:32 +01:00
return true
2015-09-09 16:49:51 +02:00
}
2017-10-30 15:10:05 +01:00
// checkServiceLabelPort checks if all service names have a port service label
// or if port container label exists for default value
func checkServiceLabelPort ( container dockerData ) error {
// If port container label is present, there is a default values for all ports, use it for the check
_ , err := strconv . Atoi ( container . Labels [ types . LabelPort ] )
if err != nil {
serviceLabelPorts := make ( map [ string ] struct { } )
serviceLabels := make ( map [ string ] struct { } )
portRegexp := regexp . MustCompile ( ` ^traefik\.(?P<service_name>.+?)\.port$ ` )
for label := range container . Labels {
// Get all port service labels
portLabel := portRegexp . FindStringSubmatch ( label )
if portLabel != nil && len ( portLabel ) > 0 {
serviceLabelPorts [ portLabel [ 0 ] ] = struct { } { }
}
// Get only one instance of all service names from service labels
servicesLabelNames := servicesPropertiesRegexp . FindStringSubmatch ( label )
if servicesLabelNames != nil && len ( servicesLabelNames ) > 0 {
serviceLabels [ strings . Split ( servicesLabelNames [ 0 ] , "." ) [ 1 ] ] = struct { } { }
}
}
// If the number of service labels is different than the number of port services label
// there is an error
if len ( serviceLabels ) == len ( serviceLabelPorts ) {
for labelPort := range serviceLabelPorts {
_ , err = strconv . Atoi ( container . Labels [ labelPort ] )
if err != nil {
break
}
}
} else {
err = errors . New ( "Port service labels missing, please use traefik.port as default value or define all port service labels" )
}
}
return err
}
2017-10-30 12:06:03 +01:00
func ( p * Provider ) getFrontendName ( container dockerData , idx int ) string {
2015-10-23 09:49:19 +02:00
// Replace '.' with '-' in quoted keys because of this issue https://github.com/BurntSushi/toml/issues/78
2017-10-30 12:06:03 +01:00
return provider . Normalize ( p . getFrontendRule ( container ) + "-" + strconv . Itoa ( idx ) )
2015-10-23 09:49:19 +02:00
}
2015-11-13 11:50:32 +01:00
// GetFrontendRule returns the frontend rule for the specified container, using
// it's label. It returns a default one (Host) if the label is not present.
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getFrontendRule ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelFrontendRule ) ; err == nil {
2015-11-13 11:50:32 +01:00
return label
}
2017-07-10 16:58:12 +02:00
if labels , err := getLabels ( container , [ ] string { labelDockerComposeProject , labelDockerComposeService } ) ; err == nil {
2017-10-23 10:33:02 +02:00
return "Host:" + getSubDomain ( labels [ labelDockerComposeService ] + "." + labels [ labelDockerComposeProject ] ) + "." + p . Domain
2017-05-03 23:24:53 +02:00
}
2017-04-07 16:30:57 +02:00
if len ( p . Domain ) > 0 {
2017-10-23 10:33:02 +02:00
return "Host:" + getSubDomain ( container . ServiceName ) + "." + p . Domain
2017-03-04 13:31:07 +01:00
}
2017-04-07 16:30:57 +02:00
return ""
2015-11-13 11:50:32 +01:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getBackend ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelBackend ) ; err == nil {
2017-04-15 15:49:53 +02:00
return provider . Normalize ( label )
2015-11-13 11:50:32 +01:00
}
2017-07-10 16:58:12 +02:00
if labels , err := getLabels ( container , [ ] string { labelDockerComposeProject , labelDockerComposeService } ) ; err == nil {
return provider . Normalize ( labels [ labelDockerComposeService ] + "_" + labels [ labelDockerComposeProject ] )
2017-03-04 13:31:07 +01:00
}
2017-04-15 15:49:53 +02:00
return provider . Normalize ( container . ServiceName )
2015-11-13 11:50:32 +01:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getIPAddress ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , labelDockerNetwork ) ; err == nil && label != "" {
2016-08-05 11:02:46 -03:00
networkSettings := container . NetworkSettings
if networkSettings . Networks != nil {
network := networkSettings . Networks [ label ]
2016-05-31 22:11:17 -07:00
if network != nil {
2016-08-05 11:02:46 -03:00
return network . Addr
2016-05-31 22:11:17 -07:00
}
2017-03-19 18:40:09 +01:00
log . Warnf ( "Could not find network named '%s' for container '%s'! Maybe you're missing the project's prefix in the label? Defaulting to first available network." , label , container . Name )
2016-05-31 22:11:17 -07:00
}
}
2016-07-05 11:44:30 +02:00
2017-10-30 14:36:04 +01:00
if container . NetworkSettings . NetworkMode . IsHost ( ) {
2017-10-25 14:20:03 -04:00
if container . Node != nil {
if container . Node . IPAddress != "" {
return container . Node . IPAddress
}
}
2016-07-05 11:44:30 +02:00
return "127.0.0.1"
}
2017-10-30 14:36:04 +01:00
if container . NetworkSettings . NetworkMode . IsContainer ( ) {
dockerClient , err := p . createClient ( )
if err != nil {
log . Warnf ( "Unable to get IP address for container %s, error: %s" , container . Name , err )
return ""
}
ctx := context . Background ( )
containerInspected , err := dockerClient . ContainerInspect ( ctx , container . NetworkSettings . NetworkMode . ConnectedContainer ( ) )
if err != nil {
log . Warnf ( "Unable to get IP address for container %s : Failed to inspect container ID %s, error: %s" , container . Name , container . NetworkSettings . NetworkMode . ConnectedContainer ( ) , err )
return ""
}
return p . getIPAddress ( parseContainer ( containerInspected ) )
}
2017-04-15 15:49:53 +02:00
if p . UseBindPortIP {
port := p . getPort ( container )
2016-09-20 14:52:35 +02:00
for netport , portBindings := range container . NetworkSettings . Ports {
if string ( netport ) == port + "/TCP" || string ( netport ) == port + "/UDP" {
for _ , p := range portBindings {
return p . HostIP
}
}
}
}
2016-07-05 11:44:30 +02:00
for _ , network := range container . NetworkSettings . Networks {
2016-08-05 11:02:46 -03:00
return network . Addr
2016-05-31 22:11:17 -07:00
}
return ""
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getPort ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelPort ) ; err == nil {
2015-11-13 11:50:32 +01:00
return label
}
2017-04-28 17:05:54 +09:00
// See iteration order in https://blog.golang.org/go-maps-in-action
var ports [ ] nat . Port
for p := range container . NetworkSettings . Ports {
ports = append ( ports , p )
}
less := func ( i , j nat . Port ) bool {
return i . Int ( ) < j . Int ( )
2015-11-13 11:50:32 +01:00
}
2017-04-28 17:05:54 +09:00
nat . Sort ( ports , less )
if len ( ports ) > 0 {
min := ports [ 0 ]
return min . Port ( )
}
2015-11-13 11:50:32 +01:00
return ""
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getWeight ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelWeight ) ; err == nil {
2015-11-13 11:50:32 +01:00
return label
}
2016-11-23 14:49:55 +01:00
return "0"
2015-11-13 11:50:32 +01:00
}
2017-10-10 11:10:02 +02:00
func ( p * Provider ) hasStickinessLabel ( container dockerData ) bool {
2017-10-12 17:50:03 +02:00
labelStickiness , errStickiness := getLabel ( container , types . LabelBackendLoadbalancerStickiness )
2017-10-16 17:38:03 +02:00
return errStickiness == nil && len ( labelStickiness ) > 0 && strings . EqualFold ( strings . TrimSpace ( labelStickiness ) , "true" )
}
2017-10-10 11:10:02 +02:00
2017-10-16 17:38:03 +02:00
func ( p * Provider ) getSticky ( container dockerData ) string {
if label , err := getLabel ( container , types . LabelBackendLoadbalancerSticky ) ; err == nil {
if len ( label ) > 0 {
log . Warnf ( "Deprecated configuration found: %s. Please use %s." , types . LabelBackendLoadbalancerSticky , types . LabelBackendLoadbalancerStickiness )
}
return label
2017-10-10 11:10:02 +02:00
}
2017-10-16 17:38:03 +02:00
return "false"
2017-10-10 11:10:02 +02:00
}
func ( p * Provider ) getStickinessCookieName ( container dockerData ) string {
if label , err := getLabel ( container , types . LabelBackendLoadbalancerStickinessCookieName ) ; err == nil {
2017-01-07 03:20:52 -05:00
return label
}
2017-10-10 11:10:02 +02:00
return ""
2017-01-07 03:20:52 -05:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getIsBackendLBSwarm ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , labelBackendLoadbalancerSwarm ) ; err == nil {
2017-01-07 03:20:52 -05:00
return label
2016-06-02 14:36:14 -04:00
}
2016-06-08 15:08:03 -04:00
return "false"
2016-06-02 14:36:14 -04:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getDomain ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelDomain ) ; err == nil {
2015-11-13 11:50:32 +01:00
return label
}
2017-04-15 15:49:53 +02:00
return p . Domain
2015-10-23 09:49:19 +02:00
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getProtocol ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if label , err := getLabel ( container , types . LabelProtocol ) ; err == nil {
2015-11-13 11:50:32 +01:00
return label
}
return "http"
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getPassHostHeader ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if passHostHeader , err := getLabel ( container , types . LabelFrontendPassHostHeader ) ; err == nil {
2015-11-13 11:50:32 +01:00
return passHostHeader
}
2016-05-10 07:43:24 -04:00
return "true"
2015-11-13 11:50:32 +01:00
}
2017-04-30 11:22:07 +02:00
func ( p * Provider ) getWhitelistSourceRange ( container dockerData ) [ ] string {
var whitelistSourceRange [ ] string
2017-07-10 16:58:12 +02:00
if whitelistSourceRangeLabel , err := getLabel ( container , types . LabelTraefikFrontendWhitelistSourceRange ) ; err == nil {
2017-04-30 11:22:07 +02:00
whitelistSourceRange = provider . SplitAndTrimString ( whitelistSourceRangeLabel )
}
return whitelistSourceRange
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getPriority ( container dockerData ) string {
2017-07-10 16:58:12 +02:00
if priority , err := getLabel ( container , types . LabelFrontendPriority ) ; err == nil {
2016-06-06 22:30:23 +02:00
return priority
}
return "0"
}
2017-04-15 15:49:53 +02:00
func ( p * Provider ) getEntryPoints ( container dockerData ) [ ] string {
2017-07-10 16:58:12 +02:00
if entryPoints , err := getLabel ( container , types . LabelFrontendEntryPoints ) ; err == nil {
2016-02-01 16:08:58 +01:00
return strings . Split ( entryPoints , "," )
}
return [ ] string { }
}
2017-04-19 11:14:05 +02:00
func ( p * Provider ) getBasicAuth ( container dockerData ) [ ] string {
2017-07-10 16:58:12 +02:00
if basicAuth , err := getLabel ( container , types . LabelFrontendAuthBasic ) ; err == nil {
2017-04-19 11:14:05 +02:00
return strings . Split ( basicAuth , "," )
}
return [ ] string { }
}
2017-10-20 10:14:03 -05:00
func ( p * Provider ) hasRequestHeaders ( container dockerData ) bool {
label , err := getLabel ( container , types . LabelFrontendRequestHeader )
return err == nil && len ( label ) > 0
}
func ( p * Provider ) hasResponseHeaders ( container dockerData ) bool {
label , err := getLabel ( container , types . LabelFrontendResponseHeader )
return err == nil && len ( label ) > 0
}
func ( p * Provider ) getRequestHeaders ( container dockerData ) map [ string ] string {
return parseCustomHeaders ( container , types . LabelFrontendRequestHeader )
}
func ( p * Provider ) getResponseHeaders ( container dockerData ) map [ string ] string {
return parseCustomHeaders ( container , types . LabelFrontendResponseHeader )
}
func parseCustomHeaders ( container dockerData , containerType string ) map [ string ] string {
customHeaders := make ( map [ string ] string )
if label , err := getLabel ( container , containerType ) ; err == nil {
for _ , headers := range strings . Split ( label , "," ) {
pair := strings . Split ( headers , ":" )
if len ( pair ) != 2 {
log . Warnf ( "Could not load header %v, skipping..." , pair )
} else {
customHeaders [ pair [ 0 ] ] = pair [ 1 ]
}
}
}
if len ( customHeaders ) == 0 {
log . Errorf ( "Could not load any custom headers" )
}
return customHeaders
}
2017-11-18 13:50:03 +01:00
func ( p * Provider ) getRedirect ( container dockerData ) string {
if entryPointredirect , err := getLabel ( container , types . LabelFrontendRedirect ) ; err == nil {
return entryPointredirect
}
return ""
}
2016-08-05 11:02:46 -03:00
func isContainerEnabled ( container dockerData , exposedByDefault bool ) bool {
2017-07-10 16:58:12 +02:00
return exposedByDefault && container . Labels [ types . LabelEnable ] != "false" || container . Labels [ types . LabelEnable ] == "true"
2016-07-14 11:32:15 +02:00
}
2016-08-05 11:02:46 -03:00
func getLabel ( container dockerData , label string ) ( string , error ) {
for key , value := range container . Labels {
2015-10-23 09:49:19 +02:00
if key == label {
return value , nil
2015-09-09 16:49:51 +02:00
}
}
2017-05-26 17:03:14 +02:00
return "" , fmt . Errorf ( "label not found: %s" , label )
2015-10-23 09:49:19 +02:00
}
2016-08-05 11:02:46 -03:00
func getLabels ( container dockerData , labels [ ] string ) ( map [ string ] string , error ) {
2015-11-05 15:14:25 +01:00
var globalErr error
2015-10-27 00:26:35 +01:00
foundLabels := map [ string ] string { }
for _ , label := range labels {
2015-11-13 11:50:32 +01:00
foundLabel , err := getLabel ( container , label )
2015-11-05 15:14:25 +01:00
// Error out only if one of them is defined.
2015-11-01 19:29:47 +01:00
if err != nil {
2017-05-26 17:03:14 +02:00
globalErr = fmt . Errorf ( "label not found: %s" , label )
2015-11-05 15:14:25 +01:00
continue
2015-10-27 00:26:35 +01:00
}
2015-11-01 19:29:47 +01:00
foundLabels [ label ] = foundLabel
2015-11-05 15:14:25 +01:00
2015-10-27 00:26:35 +01:00
}
2015-11-05 15:14:25 +01:00
return foundLabels , globalErr
2015-10-27 00:26:35 +01:00
}
2016-08-05 11:02:46 -03:00
func listContainers ( ctx context . Context , dockerClient client . ContainerAPIClient ) ( [ ] dockerData , error ) {
2016-06-08 19:39:38 +02:00
containerList , err := dockerClient . ContainerList ( ctx , dockertypes . ContainerListOptions { } )
2016-04-08 14:20:54 +02:00
if err != nil {
2016-08-05 11:02:46 -03:00
return [ ] dockerData { } , err
2016-04-08 14:20:54 +02:00
}
2016-08-05 11:02:46 -03:00
containersInspected := [ ] dockerData { }
2015-10-23 09:49:19 +02:00
2015-11-13 11:50:32 +01:00
// get inspect containers
for _ , container := range containerList {
2016-06-08 19:39:38 +02:00
containerInspected , err := dockerClient . ContainerInspect ( ctx , container . ID )
2016-04-08 14:20:54 +02:00
if err != nil {
2016-06-20 12:15:31 +02:00
log . Warnf ( "Failed to inspect container %s, error: %s" , container . ID , err )
2016-06-08 19:39:38 +02:00
} else {
2016-08-05 11:02:46 -03:00
dockerData := parseContainer ( containerInspected )
containersInspected = append ( containersInspected , dockerData )
2016-04-08 14:20:54 +02:00
}
2015-10-23 09:49:19 +02:00
}
2016-04-08 14:20:54 +02:00
return containersInspected , nil
2015-09-12 15:10:03 +02:00
}
2016-05-31 23:23:23 +02:00
2016-08-05 11:02:46 -03:00
func parseContainer ( container dockertypes . ContainerJSON ) dockerData {
dockerData := dockerData {
NetworkSettings : networkSettings { } ,
}
if container . ContainerJSONBase != nil {
dockerData . Name = container . ContainerJSONBase . Name
2017-02-02 20:18:12 -05:00
dockerData . ServiceName = dockerData . Name //Default ServiceName to be the container's Name.
2017-10-25 14:20:03 -04:00
dockerData . Node = container . ContainerJSONBase . Node
2016-08-05 11:02:46 -03:00
if container . ContainerJSONBase . HostConfig != nil {
dockerData . NetworkSettings . NetworkMode = container . ContainerJSONBase . HostConfig . NetworkMode
}
2016-11-28 16:46:37 +01:00
if container . State != nil && container . State . Health != nil {
dockerData . Health = container . State . Health . Status
}
2016-08-05 11:02:46 -03:00
}
if container . Config != nil && container . Config . Labels != nil {
dockerData . Labels = container . Config . Labels
}
if container . NetworkSettings != nil {
if container . NetworkSettings . Ports != nil {
dockerData . NetworkSettings . Ports = container . NetworkSettings . Ports
}
if container . NetworkSettings . Networks != nil {
dockerData . NetworkSettings . Networks = make ( map [ string ] * networkData )
for name , containerNetwork := range container . NetworkSettings . Networks {
dockerData . NetworkSettings . Networks [ name ] = & networkData {
ID : containerNetwork . NetworkID ,
Name : name ,
Addr : containerNetwork . IPAddress ,
}
}
}
}
return dockerData
}
2017-01-24 11:29:45 +01:00
// Escape beginning slash "/", convert all others to dash "-", and convert underscores "_" to dash "-"
2017-10-23 10:33:02 +02:00
func getSubDomain ( name string ) string {
2017-01-24 11:29:45 +01:00
return strings . Replace ( strings . Replace ( strings . TrimPrefix ( name , "/" ) , "/" , "-" , - 1 ) , "_" , "-" , - 1 )
2016-05-31 23:23:23 +02:00
}
2016-08-05 11:02:46 -03:00
2017-04-15 15:49:53 +02:00
func ( p * Provider ) listServices ( ctx context . Context , dockerClient client . APIClient ) ( [ ] dockerData , error ) {
2016-08-05 11:02:46 -03:00
serviceList , err := dockerClient . ServiceList ( ctx , dockertypes . ServiceListOptions { } )
if err != nil {
return [ ] dockerData { } , err
}
2017-10-23 10:33:02 +02:00
serverVersion , err := dockerClient . ServerVersion ( ctx )
2016-08-05 11:02:46 -03:00
networkListArgs := filters . NewArgs ( )
2017-10-23 10:33:02 +02:00
// https://docs.docker.com/engine/api/v1.29/#tag/Network (Docker 17.06)
if versions . GreaterThanOrEqualTo ( serverVersion . APIVersion , "1.29" ) {
networkListArgs . Add ( "scope" , "swarm" )
} else {
networkListArgs . Add ( "driver" , "overlay" )
}
2016-08-05 11:02:46 -03:00
networkList , err := dockerClient . NetworkList ( ctx , dockertypes . NetworkListOptions { Filters : networkListArgs } )
networkMap := make ( map [ string ] * dockertypes . NetworkResource )
if err != nil {
log . Debug ( "Failed to network inspect on client for docker, error: %s" , err )
return [ ] dockerData { } , err
}
for _ , network := range networkList {
2016-10-07 16:35:27 +02:00
networkToAdd := network
networkMap [ network . ID ] = & networkToAdd
2016-08-05 11:02:46 -03:00
}
var dockerDataList [ ] dockerData
2017-01-07 03:20:52 -05:00
var dockerDataListTasks [ ] dockerData
2016-08-05 11:02:46 -03:00
for _ , service := range serviceList {
dockerData := parseService ( service , networkMap )
2017-04-15 15:49:53 +02:00
useSwarmLB , _ := strconv . ParseBool ( p . getIsBackendLBSwarm ( dockerData ) )
2017-02-17 01:50:41 +09:00
isGlobalSvc := service . Spec . Mode . Global != nil
2016-08-05 11:02:46 -03:00
2017-01-07 03:20:52 -05:00
if useSwarmLB {
dockerDataList = append ( dockerDataList , dockerData )
} else {
2017-02-17 01:50:41 +09:00
dockerDataListTasks , err = listTasks ( ctx , dockerClient , service . ID , dockerData , networkMap , isGlobalSvc )
2017-01-07 03:20:52 -05:00
for _ , dockerDataTask := range dockerDataListTasks {
dockerDataList = append ( dockerDataList , dockerDataTask )
}
}
2016-08-05 11:02:46 -03:00
}
return dockerDataList , err
}
func parseService ( service swarmtypes . Service , networkMap map [ string ] * dockertypes . NetworkResource ) dockerData {
dockerData := dockerData {
2017-02-02 20:18:12 -05:00
ServiceName : service . Spec . Annotations . Name ,
2016-08-05 11:02:46 -03:00
Name : service . Spec . Annotations . Name ,
Labels : service . Spec . Annotations . Labels ,
NetworkSettings : networkSettings { } ,
}
if service . Spec . EndpointSpec != nil {
switch service . Spec . EndpointSpec . Mode {
2017-07-06 16:28:13 +02:00
case swarmtypes . ResolutionModeDNSRR :
2016-08-05 11:02:46 -03:00
log . Debug ( "Ignored endpoint-mode not supported, service name: %s" , dockerData . Name )
2017-07-06 16:28:13 +02:00
case swarmtypes . ResolutionModeVIP :
2016-08-05 11:02:46 -03:00
dockerData . NetworkSettings . Networks = make ( map [ string ] * networkData )
for _ , virtualIP := range service . Endpoint . VirtualIPs {
networkService := networkMap [ virtualIP . NetworkID ]
if networkService != nil {
ip , _ , _ := net . ParseCIDR ( virtualIP . Addr )
network := & networkData {
Name : networkService . Name ,
ID : virtualIP . NetworkID ,
Addr : ip . String ( ) ,
}
dockerData . NetworkSettings . Networks [ network . Name ] = network
} else {
log . Debug ( "Network not found, id: %s" , virtualIP . NetworkID )
}
2017-01-07 03:20:52 -05:00
}
}
}
return dockerData
}
func listTasks ( ctx context . Context , dockerClient client . APIClient , serviceID string ,
2017-02-17 01:50:41 +09:00
serviceDockerData dockerData , networkMap map [ string ] * dockertypes . NetworkResource , isGlobalSvc bool ) ( [ ] dockerData , error ) {
2017-01-07 03:20:52 -05:00
serviceIDFilter := filters . NewArgs ( )
serviceIDFilter . Add ( "service" , serviceID )
2017-02-17 01:50:41 +09:00
serviceIDFilter . Add ( "desired-state" , "running" )
2017-07-06 16:28:13 +02:00
taskList , err := dockerClient . TaskList ( ctx , dockertypes . TaskListOptions { Filters : serviceIDFilter } )
2017-01-07 03:20:52 -05:00
if err != nil {
return [ ] dockerData { } , err
}
var dockerDataList [ ] dockerData
for _ , task := range taskList {
2017-07-06 16:28:13 +02:00
if task . Status . State != swarmtypes . TaskStateRunning {
2017-03-16 16:38:40 +02:00
continue
}
2017-02-17 01:50:41 +09:00
dockerData := parseTasks ( task , serviceDockerData , networkMap , isGlobalSvc )
2017-01-07 03:20:52 -05:00
dockerDataList = append ( dockerDataList , dockerData )
}
return dockerDataList , err
}
2016-08-05 11:02:46 -03:00
2017-02-17 01:50:41 +09:00
func parseTasks ( task swarmtypes . Task , serviceDockerData dockerData , networkMap map [ string ] * dockertypes . NetworkResource , isGlobalSvc bool ) dockerData {
2017-01-07 03:20:52 -05:00
dockerData := dockerData {
2017-02-02 20:18:12 -05:00
ServiceName : serviceDockerData . Name ,
2017-01-07 03:20:52 -05:00
Name : serviceDockerData . Name + "." + strconv . Itoa ( task . Slot ) ,
Labels : serviceDockerData . Labels ,
NetworkSettings : networkSettings { } ,
}
2017-08-18 02:18:02 +02:00
if isGlobalSvc {
2017-02-17 01:50:41 +09:00
dockerData . Name = serviceDockerData . Name + "." + task . ID
}
2017-01-07 03:20:52 -05:00
if task . NetworksAttachments != nil {
dockerData . NetworkSettings . Networks = make ( map [ string ] * networkData )
for _ , virtualIP := range task . NetworksAttachments {
if networkService , present := networkMap [ virtualIP . Network . ID ] ; present {
// Not sure about this next loop - when would a task have multiple IP's for the same network?
for _ , addr := range virtualIP . Addresses {
ip , _ , _ := net . ParseCIDR ( addr )
network := & networkData {
ID : virtualIP . Network . ID ,
Name : networkService . Name ,
Addr : ip . String ( ) ,
}
dockerData . NetworkSettings . Networks [ network . Name ] = network
}
2016-08-05 11:02:46 -03:00
}
}
}
return dockerData
}