2015-11-01 16:35:01 +01:00
|
|
|
package provider
|
2015-09-12 15:10:03 +02:00
|
|
|
|
2015-09-08 00:15:14 +02:00
|
|
|
import (
|
2015-09-15 22:32:09 +02:00
|
|
|
"errors"
|
2015-11-01 16:35:01 +01:00
|
|
|
"fmt"
|
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"
|
2015-09-24 14:32:37 +02:00
|
|
|
log "github.com/Sirupsen/logrus"
|
2015-09-15 22:32:09 +02:00
|
|
|
"github.com/cenkalti/backoff"
|
2015-11-01 16:35:01 +01:00
|
|
|
"github.com/emilevauge/traefik/types"
|
2015-09-12 15:10:03 +02:00
|
|
|
"github.com/fsouza/go-dockerclient"
|
2015-09-07 10:38:58 +02:00
|
|
|
)
|
2015-09-09 22:39:08 +02:00
|
|
|
|
2015-11-01 19:29:47 +01:00
|
|
|
// Docker holds configurations of the Docker provider.
|
2015-11-02 19:48:34 +01:00
|
|
|
type Docker struct {
|
2016-01-13 22:46:44 +01:00
|
|
|
BaseProvider `mapstructure:",squash"`
|
|
|
|
Endpoint string
|
|
|
|
Domain string
|
|
|
|
TLS *DockerTLS
|
2015-11-20 23:05:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// DockerTLS holds TLS specific configurations
|
|
|
|
type DockerTLS struct {
|
|
|
|
CA string
|
|
|
|
Cert string
|
|
|
|
Key string
|
|
|
|
InsecureSkipVerify bool
|
2015-09-09 22:39:08 +02:00
|
|
|
}
|
|
|
|
|
2015-11-01 19:29:47 +01:00
|
|
|
// Provide allows the provider to provide configurations to traefik
|
|
|
|
// using the given configuration channel.
|
2015-11-02 19:48:34 +01:00
|
|
|
func (provider *Docker) Provide(configurationChan chan<- types.ConfigMessage) error {
|
2015-11-01 19:29:47 +01:00
|
|
|
|
2015-11-20 23:05:06 +08:00
|
|
|
var dockerClient *docker.Client
|
|
|
|
var err error
|
|
|
|
|
|
|
|
if provider.TLS != nil {
|
|
|
|
dockerClient, err = docker.NewTLSClient(provider.Endpoint,
|
|
|
|
provider.TLS.Cert, provider.TLS.Key, provider.TLS.CA)
|
|
|
|
if err == nil {
|
|
|
|
dockerClient.TLSConfig.InsecureSkipVerify = provider.TLS.InsecureSkipVerify
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
dockerClient, err = docker.NewClient(provider.Endpoint)
|
|
|
|
}
|
2015-11-01 19:29:47 +01:00
|
|
|
if err != nil {
|
2015-10-01 12:04:25 +02:00
|
|
|
log.Errorf("Failed to create a client for docker, error: %s", err)
|
|
|
|
return err
|
2015-11-01 19:29:47 +01:00
|
|
|
}
|
|
|
|
err = dockerClient.Ping()
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Docker connection error %+v", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
log.Debug("Docker connection established")
|
|
|
|
if provider.Watch {
|
|
|
|
dockerEvents := make(chan *docker.APIEvents)
|
|
|
|
dockerClient.AddEventListener(dockerEvents)
|
|
|
|
log.Debug("Docker listening")
|
|
|
|
go func() {
|
|
|
|
operation := func() error {
|
|
|
|
for {
|
|
|
|
event := <-dockerEvents
|
|
|
|
if event == nil {
|
|
|
|
return errors.New("Docker event nil")
|
|
|
|
// log.Fatalf("Docker connection error")
|
|
|
|
}
|
|
|
|
if event.Status == "start" || event.Status == "die" {
|
|
|
|
log.Debugf("Docker event receveived %+v", event)
|
2015-11-13 11:50:32 +01:00
|
|
|
configuration := provider.loadDockerConfig(listContainers(dockerClient))
|
2015-11-01 19:29:47 +01:00
|
|
|
if configuration != nil {
|
2015-11-13 11:50:32 +01:00
|
|
|
configurationChan <- types.ConfigMessage{
|
|
|
|
ProviderName: "docker",
|
|
|
|
Configuration: configuration,
|
|
|
|
}
|
2015-09-10 22:54:37 +02:00
|
|
|
}
|
2015-09-10 09:06:37 +02:00
|
|
|
}
|
2015-09-09 22:39:08 +02:00
|
|
|
}
|
2015-11-01 19:29:47 +01:00
|
|
|
}
|
|
|
|
notify := func(err error, time time.Duration) {
|
|
|
|
log.Errorf("Docker connection error %+v, retrying in %s", err, time)
|
|
|
|
}
|
|
|
|
err := backoff.RetryNotify(operation, backoff.NewExponentialBackOff(), notify)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Cannot connect to docker server %+v", err)
|
|
|
|
}
|
|
|
|
}()
|
2015-09-09 22:39:08 +02:00
|
|
|
}
|
2015-11-01 19:29:47 +01:00
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
configuration := provider.loadDockerConfig(listContainers(dockerClient))
|
|
|
|
configurationChan <- types.ConfigMessage{
|
|
|
|
ProviderName: "docker",
|
|
|
|
Configuration: configuration,
|
|
|
|
}
|
2015-10-01 12:04:25 +02:00
|
|
|
return nil
|
2015-09-07 10:38:58 +02:00
|
|
|
}
|
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
func (provider *Docker) loadDockerConfig(containersInspected []docker.Container) *types.Configuration {
|
2015-10-08 21:21:51 +02:00
|
|
|
var DockerFuncMap = template.FuncMap{
|
2015-11-13 11:50:32 +01:00
|
|
|
"getBackend": provider.getBackend,
|
|
|
|
"getPort": provider.getPort,
|
|
|
|
"getWeight": provider.getWeight,
|
|
|
|
"getDomain": provider.getDomain,
|
|
|
|
"getProtocol": provider.getProtocol,
|
|
|
|
"getPassHostHeader": provider.getPassHostHeader,
|
|
|
|
"getFrontendValue": provider.getFrontendValue,
|
|
|
|
"getFrontendRule": provider.getFrontendRule,
|
|
|
|
"replace": replace,
|
2015-09-10 22:54:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// filter containers
|
2015-11-13 11:50:32 +01:00
|
|
|
filteredContainers := fun.Filter(containerFilter, containersInspected).([]docker.Container)
|
2015-09-10 22:54:37 +02:00
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
frontends := map[string][]docker.Container{}
|
2015-09-10 22:54:37 +02:00
|
|
|
for _, container := range filteredContainers {
|
2015-10-23 09:49:19 +02:00
|
|
|
frontends[provider.getFrontendName(container)] = append(frontends[provider.getFrontendName(container)], container)
|
2015-09-07 10:38:58 +02:00
|
|
|
}
|
2015-09-09 16:49:51 +02:00
|
|
|
|
|
|
|
templateObjects := struct {
|
2015-09-07 10:38:58 +02:00
|
|
|
Containers []docker.Container
|
2015-10-23 09:49:19 +02:00
|
|
|
Frontends map[string][]docker.Container
|
2015-09-09 17:50:02 +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,
|
2015-09-09 17:10:43 +02:00
|
|
|
provider.Domain,
|
2015-09-07 10:38:58 +02:00
|
|
|
}
|
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
configuration, err := provider.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
|
|
|
}
|
2015-11-13 11:50:32 +01:00
|
|
|
return configuration
|
|
|
|
}
|
2015-09-07 10:38:58 +02:00
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
func containerFilter(container docker.Container) bool {
|
|
|
|
if len(container.NetworkSettings.Ports) == 0 {
|
|
|
|
log.Debugf("Filtering container without port %s", container.Name)
|
|
|
|
return false
|
2015-09-07 10:38:58 +02:00
|
|
|
}
|
2015-11-13 11:50:32 +01:00
|
|
|
_, err := strconv.Atoi(container.Config.Labels["traefik.port"])
|
|
|
|
if len(container.NetworkSettings.Ports) > 1 && err != nil {
|
|
|
|
log.Debugf("Filtering container with more than 1 port and no traefik.port label %s", container.Name)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if container.Config.Labels["traefik.enable"] == "false" {
|
|
|
|
log.Debugf("Filtering disabled container %s", container.Name)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
labels, err := getLabels(container, []string{"traefik.frontend.rule", "traefik.frontend.value"})
|
|
|
|
if len(labels) != 0 && err != nil {
|
|
|
|
log.Debugf("Filtering bad labeled container %s", container.Name)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
2015-09-09 16:49:51 +02:00
|
|
|
}
|
|
|
|
|
2015-11-02 19:48:34 +01:00
|
|
|
func (provider *Docker) getFrontendName(container docker.Container) 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
|
2015-11-13 11:50:32 +01:00
|
|
|
frontendName := fmt.Sprintf("%s-%s", provider.getFrontendRule(container), provider.getFrontendValue(container))
|
2015-11-12 17:04:19 -07:00
|
|
|
frontendName = strings.Replace(frontendName, "[", "", -1)
|
|
|
|
frontendName = strings.Replace(frontendName, "]", "", -1)
|
|
|
|
|
2015-10-27 00:26:35 +01:00
|
|
|
return strings.Replace(frontendName, ".", "-", -1)
|
2015-10-23 09:49:19 +02:00
|
|
|
}
|
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
// GetFrontendValue returns the frontend value for the specified container, using
|
|
|
|
// it's label. It returns a default one if the label is not present.
|
|
|
|
func (provider *Docker) getFrontendValue(container docker.Container) string {
|
|
|
|
if label, err := getLabel(container, "traefik.frontend.value"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return getEscapedName(container.Name) + "." + provider.Domain
|
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.
|
|
|
|
func (provider *Docker) getFrontendRule(container docker.Container) string {
|
|
|
|
if label, err := getLabel(container, "traefik.frontend.rule"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return "Host"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (provider *Docker) getBackend(container docker.Container) string {
|
|
|
|
if label, err := getLabel(container, "traefik.backend"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return getEscapedName(container.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (provider *Docker) getPort(container docker.Container) string {
|
|
|
|
if label, err := getLabel(container, "traefik.port"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
for key := range container.NetworkSettings.Ports {
|
|
|
|
return key.Port()
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (provider *Docker) getWeight(container docker.Container) string {
|
|
|
|
if label, err := getLabel(container, "traefik.weight"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return "0"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (provider *Docker) getDomain(container docker.Container) string {
|
|
|
|
if label, err := getLabel(container, "traefik.domain"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return provider.Domain
|
2015-10-23 09:49:19 +02:00
|
|
|
}
|
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
func (provider *Docker) getProtocol(container docker.Container) string {
|
|
|
|
if label, err := getLabel(container, "traefik.protocol"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return "http"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (provider *Docker) getPassHostHeader(container docker.Container) string {
|
|
|
|
if passHostHeader, err := getLabel(container, "traefik.frontend.passHostHeader"); err == nil {
|
|
|
|
return passHostHeader
|
|
|
|
}
|
|
|
|
return "false"
|
|
|
|
}
|
|
|
|
|
|
|
|
func getLabel(container docker.Container, label string) (string, error) {
|
2015-09-09 16:49:51 +02:00
|
|
|
for key, value := range container.Config.Labels {
|
2015-10-23 09:49:19 +02:00
|
|
|
if key == label {
|
|
|
|
return value, nil
|
2015-09-09 16:49:51 +02:00
|
|
|
}
|
|
|
|
}
|
2015-10-23 09:49:19 +02:00
|
|
|
return "", errors.New("Label not found:" + label)
|
|
|
|
}
|
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
func getLabels(container docker.Container, 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 {
|
2015-11-05 15:14:25 +01:00
|
|
|
globalErr = errors.New("Label not found: " + label)
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
func listContainers(dockerClient *docker.Client) []docker.Container {
|
|
|
|
containerList, _ := dockerClient.ListContainers(docker.ListContainersOptions{})
|
|
|
|
containersInspected := []docker.Container{}
|
2015-10-23 09:49:19 +02:00
|
|
|
|
2015-11-13 11:50:32 +01:00
|
|
|
// get inspect containers
|
|
|
|
for _, container := range containerList {
|
|
|
|
containerInspected, _ := dockerClient.InspectContainer(container.ID)
|
|
|
|
containersInspected = append(containersInspected, *containerInspected)
|
2015-10-23 09:49:19 +02:00
|
|
|
}
|
2015-11-13 11:50:32 +01:00
|
|
|
return containersInspected
|
2015-09-12 15:10:03 +02:00
|
|
|
}
|