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"
2018-01-15 08:26:03 -05:00
"io"
2016-08-05 11:02:46 -03:00
"net"
2016-04-08 14:20:54 +02:00
"net/http"
2015-09-24 17:16:13 +02:00
"strconv"
"strings"
"time"
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"
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
)
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" `
2018-06-13 14:50:04 +02:00
Network string ` description:"Default Docker network used" 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
2018-03-23 13:30:03 +01:00
SegmentLabels map [ string ] string
SegmentName string
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-12-04 11:40:03 +01:00
func ( p * Provider ) createClient ( ) ( client . APIClient , error ) {
2016-04-08 14:20:54 +02:00
var httpClient * http . Client
2017-11-28 11:16:03 +01: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 ,
}
2018-03-23 13:30:03 +01:00
hostURL , err := client . ParseHostURL ( p . Endpoint )
2016-04-08 14:20:54 +02:00
if err != nil {
return nil , err
}
2018-03-23 13:30:03 +01:00
sockets . ConfigureTransport ( tr , hostURL . Scheme , hostURL . Host )
2016-04-08 14:20:54 +02:00
httpClient = & http . Client {
Transport : tr ,
}
2017-11-28 11:16:03 +01:00
}
2016-06-27 16:14:56 +02:00
2017-11-28 11:16:03 +01:00
httpHeaders := map [ string ] string {
"User-Agent" : "Traefik " + version . Version ,
2016-04-08 14:20:54 +02:00
}
2017-11-28 11:16:03 +01:00
var apiVersion string
2017-04-15 15:49:53 +02:00
if p . SwarmMode {
2017-11-28 11:16:03 +01:00
apiVersion = SwarmAPIVersion
2016-08-05 11:02:46 -03:00
} else {
2017-11-28 11:16:03 +01:00
apiVersion = DockerAPIVersion
2016-08-05 11:02:46 -03:00
}
2017-11-28 11:16:03 +01:00
return client . NewClient ( p . Endpoint , apiVersion , httpClient , httpHeaders )
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 ( )
2017-11-28 11:16:03 +01:00
serverVersion , 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-11-28 11:16:03 +01:00
log . Debugf ( "Provider connection established with docker %s (API %s)" , serverVersion . Version , serverVersion . APIVersion )
2016-08-05 11:02:46 -03:00
var dockerDataList [ ] dockerData
2017-04-15 15:49:53 +02:00
if p . SwarmMode {
2017-11-28 11:16:03 +01:00
dockerDataList , err = 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-12-02 19:26:44 +01:00
configuration := p . buildConfiguration ( 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-11-28 11:16:03 +01:00
services , err := 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-12-02 19:26:44 +01:00
configuration := p . buildConfiguration ( 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 ) {
2017-12-06 10:52:03 +01:00
<- stop
cancel ( )
2016-08-05 11:02:46 -03:00
} )
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-12-02 19:26:44 +01:00
configuration := p . buildConfiguration ( 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 )
2018-01-15 08:26:03 -05:00
for {
select {
case event := <- eventsc :
if event . Action == "start" ||
event . Action == "die" ||
strings . HasPrefix ( event . Action , "health_status" ) {
startStopHandle ( event )
}
case err := <- errc :
if err == io . EOF {
log . Debug ( "Provider event stream closed" )
}
return err
2017-07-06 16:28:13 +02:00
}
}
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
}
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 {
2017-12-02 19:26:44 +01:00
return nil , err
2016-04-08 14:20:54 +02:00
}
2015-10-23 09:49:19 +02:00
2017-11-28 11:16:03 +01:00
var containersInspected [ ] dockerData
2015-11-13 11:50:32 +01:00
// get inspect containers
for _ , container := range containerList {
2018-02-12 17:50:05 +01:00
dData := inspectContainers ( ctx , dockerClient , container . ID )
if len ( dData . Name ) > 0 {
containersInspected = append ( containersInspected , dData )
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
2018-02-12 17:50:05 +01:00
func inspectContainers ( ctx context . Context , dockerClient client . ContainerAPIClient , containerID string ) dockerData {
dData := dockerData { }
containerInspected , err := dockerClient . ContainerInspect ( ctx , containerID )
if err != nil {
log . Warnf ( "Failed to inspect container %s, error: %s" , containerID , err )
} else {
// This condition is here to avoid to have empty IP https://github.com/containous/traefik/issues/2459
// We register only container which are running
if containerInspected . ContainerJSONBase != nil && containerInspected . ContainerJSONBase . State != nil && containerInspected . ContainerJSONBase . State . Running {
dData = parseContainer ( containerInspected )
}
}
return dData
}
2016-08-05 11:02:46 -03:00
func parseContainer ( container dockertypes . ContainerJSON ) dockerData {
2017-12-02 19:26:44 +01:00
dData := dockerData {
2016-08-05 11:02:46 -03:00
NetworkSettings : networkSettings { } ,
}
if container . ContainerJSONBase != nil {
2017-12-02 19:26:44 +01:00
dData . Name = container . ContainerJSONBase . Name
2018-03-23 13:30:03 +01:00
dData . ServiceName = dData . Name // Default ServiceName to be the container's Name.
2017-12-02 19:26:44 +01:00
dData . Node = container . ContainerJSONBase . Node
2016-08-05 11:02:46 -03:00
if container . ContainerJSONBase . HostConfig != nil {
2017-12-02 19:26:44 +01:00
dData . NetworkSettings . NetworkMode = container . ContainerJSONBase . HostConfig . NetworkMode
2016-08-05 11:02:46 -03:00
}
2016-11-28 16:46:37 +01:00
if container . State != nil && container . State . Health != nil {
2017-12-02 19:26:44 +01:00
dData . Health = container . State . Health . Status
2016-11-28 16:46:37 +01:00
}
2016-08-05 11:02:46 -03:00
}
if container . Config != nil && container . Config . Labels != nil {
2017-12-02 19:26:44 +01:00
dData . Labels = container . Config . Labels
2016-08-05 11:02:46 -03:00
}
if container . NetworkSettings != nil {
if container . NetworkSettings . Ports != nil {
2017-12-02 19:26:44 +01:00
dData . NetworkSettings . Ports = container . NetworkSettings . Ports
2016-08-05 11:02:46 -03:00
}
if container . NetworkSettings . Networks != nil {
2017-12-02 19:26:44 +01:00
dData . NetworkSettings . Networks = make ( map [ string ] * networkData )
2016-08-05 11:02:46 -03:00
for name , containerNetwork := range container . NetworkSettings . Networks {
2017-12-02 19:26:44 +01:00
dData . NetworkSettings . Networks [ name ] = & networkData {
2016-08-05 11:02:46 -03:00
ID : containerNetwork . NetworkID ,
Name : name ,
Addr : containerNetwork . IPAddress ,
}
}
}
}
2017-12-02 19:26:44 +01:00
return dData
2016-05-31 23:23:23 +02:00
}
2016-08-05 11:02:46 -03:00
2017-11-28 11:16:03 +01:00
func 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 {
2017-12-02 19:26:44 +01:00
return nil , err
2016-08-05 11:02:46 -03:00
}
2017-10-23 10:33:02 +02:00
serverVersion , err := dockerClient . ServerVersion ( ctx )
2017-12-02 19:26:44 +01:00
if err != nil {
return nil , err
}
2017-10-23 10:33:02 +02:00
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 } )
if err != nil {
2017-12-01 14:34:03 +01:00
log . Debugf ( "Failed to network inspect on client for docker, error: %s" , err )
2017-12-02 19:26:44 +01:00
return nil , err
2016-08-05 11:02:46 -03:00
}
2017-12-02 19:26:44 +01:00
networkMap := make ( map [ string ] * dockertypes . NetworkResource )
2016-08-05 11:02:46 -03:00
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 {
2017-12-02 19:26:44 +01:00
dData := parseService ( service , networkMap )
2016-08-05 11:02:46 -03:00
2018-02-07 15:24:44 +01:00
if isBackendLBSwarm ( dData ) {
if len ( dData . NetworkSettings . Networks ) > 0 {
2017-12-02 19:26:44 +01:00
dockerDataList = append ( dockerDataList , dData )
2018-02-05 11:34:03 +01:00
}
} else {
isGlobalSvc := service . Spec . Mode . Global != nil
2018-02-07 15:24:44 +01:00
dockerDataListTasks , err = listTasks ( ctx , dockerClient , service . ID , dData , networkMap , isGlobalSvc )
if err != nil {
log . Warn ( err )
2017-12-01 14:34:03 +01:00
} else {
2018-02-07 15:24:44 +01:00
dockerDataList = append ( dockerDataList , dockerDataListTasks ... )
2017-01-07 03:20:52 -05:00
}
}
2016-08-05 11:02:46 -03:00
}
return dockerDataList , err
}
func parseService ( service swarmtypes . Service , networkMap map [ string ] * dockertypes . NetworkResource ) dockerData {
2017-12-02 19:26:44 +01:00
dData := 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 {
2017-12-01 14:34:03 +01:00
if service . Spec . EndpointSpec . Mode == swarmtypes . ResolutionModeDNSRR {
2018-02-07 15:24:44 +01:00
if isBackendLBSwarm ( dData ) {
2018-02-05 11:34:03 +01:00
log . Warnf ( "Ignored %s endpoint-mode not supported, service name: %s. Fallback to Træfik load balancing" , swarmtypes . ResolutionModeDNSRR , service . Spec . Annotations . Name )
}
2017-12-01 14:34:03 +01:00
} else if service . Spec . EndpointSpec . Mode == swarmtypes . ResolutionModeVIP {
2017-12-02 19:26:44 +01:00
dData . NetworkSettings . Networks = make ( map [ string ] * networkData )
2016-08-05 11:02:46 -03:00
for _ , virtualIP := range service . Endpoint . VirtualIPs {
networkService := networkMap [ virtualIP . NetworkID ]
if networkService != nil {
2018-02-12 17:50:05 +01:00
if len ( virtualIP . Addr ) > 0 {
ip , _ , _ := net . ParseCIDR ( virtualIP . Addr )
network := & networkData {
Name : networkService . Name ,
ID : virtualIP . NetworkID ,
Addr : ip . String ( ) ,
}
dData . NetworkSettings . Networks [ network . Name ] = network
} else {
log . Debugf ( "No virtual IPs found in network %s" , virtualIP . NetworkID )
2016-08-05 11:02:46 -03:00
}
} else {
2017-12-01 14:34:03 +01:00
log . Debugf ( "Network not found, id: %s" , virtualIP . NetworkID )
2016-08-05 11:02:46 -03:00
}
2017-01-07 03:20:52 -05:00
}
}
}
2017-12-02 19:26:44 +01:00
return dData
2017-01-07 03:20:52 -05:00
}
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-01-07 03:20:52 -05:00
2017-12-02 19:26:44 +01:00
taskList , err := dockerClient . TaskList ( ctx , dockertypes . TaskListOptions { Filters : serviceIDFilter } )
2017-01-07 03:20:52 -05:00
if err != nil {
2017-12-02 19:26:44 +01:00
return nil , err
2017-01-07 03:20:52 -05:00
}
2017-12-02 19:26:44 +01:00
var dockerDataList [ ] dockerData
2017-01-07 03:20:52 -05:00
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-12-02 19:26:44 +01:00
dData := parseTasks ( task , serviceDockerData , networkMap , isGlobalSvc )
2018-02-12 17:50:05 +01:00
if len ( dData . NetworkSettings . Networks ) > 0 {
dockerDataList = append ( dockerDataList , dData )
}
2017-01-07 03:20:52 -05:00
}
return dockerDataList , err
}
2016-08-05 11:02:46 -03:00
2018-02-12 17:50:05 +01:00
func parseTasks ( task swarmtypes . Task , serviceDockerData dockerData ,
networkMap map [ string ] * dockertypes . NetworkResource , isGlobalSvc bool ) dockerData {
2017-12-02 19:26:44 +01:00
dData := 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-12-02 19:26:44 +01:00
dData . Name = serviceDockerData . Name + "." + task . ID
2017-02-17 01:50:41 +09:00
}
2017-01-07 03:20:52 -05:00
if task . NetworksAttachments != nil {
2017-12-02 19:26:44 +01:00
dData . NetworkSettings . Networks = make ( map [ string ] * networkData )
2017-01-07 03:20:52 -05:00
for _ , virtualIP := range task . NetworksAttachments {
if networkService , present := networkMap [ virtualIP . Network . ID ] ; present {
2018-02-12 17:50:05 +01:00
if len ( virtualIP . Addresses ) > 0 {
// 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 ( ) ,
}
dData . NetworkSettings . Networks [ network . Name ] = network
2017-01-07 03:20:52 -05:00
}
2018-02-12 17:50:05 +01:00
} else {
log . Debugf ( "No IP addresses found for network %s" , virtualIP . Network . ID )
2017-01-07 03:20:52 -05:00
}
2016-08-05 11:02:46 -03:00
}
}
}
2017-12-02 19:26:44 +01:00
return dData
2016-08-05 11:02:46 -03:00
}