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"`
|
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-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,
|
|
|
|
}
|
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,
|
|
|
|
}
|
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 {
|
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 {
|
2017-11-28 13:58:04 +01:00
|
|
|
// 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 {
|
2017-12-02 19:26:44 +01:00
|
|
|
dData := parseContainer(containerInspected)
|
|
|
|
containersInspected = append(containersInspected, dData)
|
2017-11-28 13:58:04 +01:00
|
|
|
}
|
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 {
|
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
|
|
|
|
dData.ServiceName = dData.Name //Default ServiceName to be the container's Name.
|
|
|
|
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)
|
|
|
|
if len(dData.NetworkSettings.Networks) > 0 {
|
|
|
|
useSwarmLB := isBackendLBSwarm(dData)
|
2016-08-05 11:02:46 -03:00
|
|
|
|
2017-12-01 14:34:03 +01:00
|
|
|
if useSwarmLB {
|
2017-12-02 19:26:44 +01:00
|
|
|
dockerDataList = append(dockerDataList, dData)
|
2017-12-01 14:34:03 +01:00
|
|
|
} else {
|
|
|
|
isGlobalSvc := service.Spec.Mode.Global != nil
|
2017-01-07 03:20:52 -05:00
|
|
|
|
2017-12-02 19:26:44 +01:00
|
|
|
dockerDataListTasks, err = listTasks(ctx, dockerClient, service.ID, dData, networkMap, isGlobalSvc)
|
|
|
|
if err != nil {
|
|
|
|
log.Warn(err)
|
|
|
|
} else {
|
|
|
|
dockerDataList = append(dockerDataList, dockerDataListTasks...)
|
2017-12-01 14:34:03 +01:00
|
|
|
}
|
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-02 19:26:44 +01:00
|
|
|
switch service.Spec.EndpointSpec.Mode {
|
|
|
|
case swarmtypes.ResolutionModeDNSRR:
|
2017-12-01 14:34:03 +01:00
|
|
|
log.Warnf("Ignored endpoint-mode not supported, service name: %s", service.Spec.Annotations.Name)
|
2017-12-02 19:26:44 +01:00
|
|
|
case swarmtypes.ResolutionModeVIP:
|
|
|
|
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 {
|
|
|
|
ip, _, _ := net.ParseCIDR(virtualIP.Addr)
|
|
|
|
network := &networkData{
|
|
|
|
Name: networkService.Name,
|
|
|
|
ID: virtualIP.NetworkID,
|
|
|
|
Addr: ip.String(),
|
|
|
|
}
|
2017-12-02 19:26:44 +01:00
|
|
|
dData.NetworkSettings.Networks[network.Name] = network
|
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)
|
|
|
|
dockerDataList = append(dockerDataList, dData)
|
2017-01-07 03:20:52 -05:00
|
|
|
}
|
|
|
|
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-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 {
|
|
|
|
// 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(),
|
|
|
|
}
|
2017-12-02 19:26:44 +01:00
|
|
|
dData.NetworkSettings.Networks[network.Name] = network
|
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
|
|
|
}
|