2015-11-01 16:35:01 +01:00
|
|
|
package provider
|
2015-09-12 15:10:03 +02:00
|
|
|
|
2015-09-09 22:39:08 +02:00
|
|
|
import (
|
|
|
|
"bytes"
|
2015-11-01 16:35:01 +01:00
|
|
|
"errors"
|
2015-09-24 17:16:13 +02:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"text/template"
|
|
|
|
|
2015-09-09 22:39:08 +02:00
|
|
|
"github.com/BurntSushi/toml"
|
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-11-01 16:35:01 +01:00
|
|
|
"github.com/emilevauge/traefik/autogen"
|
|
|
|
"github.com/emilevauge/traefik/types"
|
2015-09-12 15:10:03 +02:00
|
|
|
"github.com/gambol99/go-marathon"
|
2015-09-09 22:39:08 +02:00
|
|
|
)
|
|
|
|
|
2015-11-01 19:29:47 +01:00
|
|
|
// Marathon holds configuration of the Marathon provider.
|
2015-11-02 19:48:34 +01:00
|
|
|
type Marathon struct {
|
2015-09-10 15:13:35 +02:00
|
|
|
Watch bool
|
|
|
|
Endpoint string
|
|
|
|
Domain string
|
|
|
|
Filename string
|
|
|
|
NetworkInterface string
|
2015-11-01 19:29:47 +01:00
|
|
|
marathonClient marathon.Marathon
|
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 *Marathon) Provide(configurationChan chan<- types.ConfigMessage) error {
|
2015-09-09 22:39:08 +02:00
|
|
|
config := marathon.NewDefaultConfig()
|
|
|
|
config.URL = provider.Endpoint
|
2015-09-10 15:13:35 +02:00
|
|
|
config.EventsInterface = provider.NetworkInterface
|
2015-09-24 17:16:13 +02:00
|
|
|
client, err := marathon.NewClient(config)
|
|
|
|
if err != nil {
|
2015-09-24 14:32:37 +02:00
|
|
|
log.Errorf("Failed to create a client for marathon, error: %s", err)
|
2015-10-01 12:04:25 +02:00
|
|
|
return err
|
2015-09-24 17:16:13 +02:00
|
|
|
}
|
|
|
|
provider.marathonClient = client
|
|
|
|
update := make(marathon.EventsChannel, 5)
|
|
|
|
if provider.Watch {
|
|
|
|
if err := client.AddEventsListener(update, marathon.EVENTS_APPLICATIONS); err != nil {
|
|
|
|
log.Errorf("Failed to register for subscriptions, %s", err)
|
|
|
|
} else {
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
event := <-update
|
|
|
|
log.Debug("Marathon event receveived", event)
|
|
|
|
configuration := provider.loadMarathonConfig()
|
|
|
|
if configuration != nil {
|
2015-11-01 16:35:01 +01:00
|
|
|
configurationChan <- types.ConfigMessage{"marathon", configuration}
|
2015-09-09 23:09:16 +02:00
|
|
|
}
|
2015-09-24 17:16:13 +02:00
|
|
|
}
|
|
|
|
}()
|
2015-09-09 23:09:16 +02:00
|
|
|
}
|
2015-09-09 22:39:08 +02:00
|
|
|
}
|
2015-09-24 17:16:13 +02:00
|
|
|
|
|
|
|
configuration := provider.loadMarathonConfig()
|
2015-11-01 16:35:01 +01:00
|
|
|
configurationChan <- types.ConfigMessage{"marathon", configuration}
|
2015-10-01 12:04:25 +02:00
|
|
|
return nil
|
2015-09-09 22:39:08 +02:00
|
|
|
}
|
|
|
|
|
2015-11-02 19:48:34 +01:00
|
|
|
func (provider *Marathon) loadMarathonConfig() *types.Configuration {
|
2015-10-08 21:21:51 +02:00
|
|
|
var MarathonFuncMap = template.FuncMap{
|
|
|
|
"getPort": func(task marathon.Task) string {
|
|
|
|
for _, port := range task.Ports {
|
|
|
|
return strconv.Itoa(port)
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
},
|
2015-10-23 09:49:19 +02:00
|
|
|
"getWeight": func(task marathon.Task, applications []marathon.Application) string {
|
2015-10-27 00:26:35 +01:00
|
|
|
application, errApp := getApplication(task, applications)
|
|
|
|
if errApp != nil {
|
2015-10-23 09:49:19 +02:00
|
|
|
log.Errorf("Unable to get marathon application from task %s", task.AppID)
|
|
|
|
return "0"
|
2015-10-08 21:21:51 +02:00
|
|
|
}
|
2015-10-27 00:26:35 +01:00
|
|
|
if label, err := provider.getLabel(application, "traefik.weight"); err == nil {
|
2015-10-23 09:49:19 +02:00
|
|
|
return label
|
2015-10-08 21:21:51 +02:00
|
|
|
}
|
|
|
|
return "0"
|
|
|
|
},
|
|
|
|
"getDomain": func(application marathon.Application) string {
|
2015-10-23 09:49:19 +02:00
|
|
|
if label, err := provider.getLabel(application, "traefik.domain"); err == nil {
|
|
|
|
return label
|
2015-10-08 21:21:51 +02:00
|
|
|
}
|
|
|
|
return provider.Domain
|
|
|
|
},
|
|
|
|
"replace": func(s1 string, s2 string, s3 string) string {
|
|
|
|
return strings.Replace(s3, s1, s2, -1)
|
|
|
|
},
|
2015-10-23 18:59:08 +02:00
|
|
|
"getProtocol": func(task marathon.Task, applications []marathon.Application) string {
|
2015-10-27 00:26:35 +01:00
|
|
|
application, errApp := getApplication(task, applications)
|
|
|
|
if errApp != nil {
|
2015-10-23 09:49:19 +02:00
|
|
|
log.Errorf("Unable to get marathon application from task %s", task.AppID)
|
|
|
|
return "http"
|
|
|
|
}
|
2015-10-27 00:26:35 +01:00
|
|
|
if label, err := provider.getLabel(application, "traefik.protocol"); err == nil {
|
2015-10-23 09:49:19 +02:00
|
|
|
return label
|
|
|
|
}
|
|
|
|
return "http"
|
|
|
|
},
|
2015-10-30 11:33:41 +01:00
|
|
|
"getPassHostHeader": func(application marathon.Application) string {
|
|
|
|
if passHostHeader, err := provider.getLabel(application, "traefik.frontend.passHostHeader"); err == nil {
|
|
|
|
return passHostHeader
|
|
|
|
}
|
|
|
|
return "false"
|
|
|
|
},
|
2015-10-23 09:49:19 +02:00
|
|
|
"getFrontendValue": provider.GetFrontendValue,
|
|
|
|
"getFrontendRule": provider.GetFrontendRule,
|
2015-10-08 21:21:51 +02:00
|
|
|
}
|
2015-11-01 16:35:01 +01:00
|
|
|
configuration := new(types.Configuration)
|
2015-09-09 22:39:08 +02:00
|
|
|
|
|
|
|
applications, err := provider.marathonClient.Applications(nil)
|
2015-09-12 15:10:03 +02:00
|
|
|
if err != nil {
|
2015-09-24 14:32:37 +02:00
|
|
|
log.Errorf("Failed to create a client for marathon, error: %s", err)
|
2015-09-09 22:39:08 +02:00
|
|
|
return nil
|
|
|
|
}
|
2015-09-10 22:54:37 +02:00
|
|
|
|
2015-09-09 22:39:08 +02:00
|
|
|
tasks, err := provider.marathonClient.AllTasks()
|
2015-09-12 15:10:03 +02:00
|
|
|
if err != nil {
|
2015-09-24 14:32:37 +02:00
|
|
|
log.Errorf("Failed to create a client for marathon, error: %s", err)
|
2015-09-09 22:39:08 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-09-10 22:54:37 +02:00
|
|
|
//filter tasks
|
|
|
|
filteredTasks := fun.Filter(func(task marathon.Task) bool {
|
2015-09-12 15:10:03 +02:00
|
|
|
if len(task.Ports) == 0 {
|
2015-10-30 00:15:03 +01:00
|
|
|
log.Debug("Filtering marathon task without port %s", task.AppID)
|
2015-09-10 22:54:37 +02:00
|
|
|
return false
|
|
|
|
}
|
2015-10-27 00:26:35 +01:00
|
|
|
application, errApp := getApplication(task, applications.Apps)
|
|
|
|
if errApp != nil {
|
2015-10-08 09:31:30 +02:00
|
|
|
log.Errorf("Unable to get marathon application from task %s", task.AppID)
|
|
|
|
return false
|
|
|
|
}
|
2015-09-10 22:54:37 +02:00
|
|
|
_, err := strconv.Atoi(application.Labels["traefik.port"])
|
2015-09-12 15:10:03 +02:00
|
|
|
if len(application.Ports) > 1 && err != nil {
|
2015-10-30 00:15:03 +01:00
|
|
|
log.Debugf("Filtering marathon task %s with more than 1 port and no traefik.port label", task.AppID)
|
2015-09-10 22:54:37 +02:00
|
|
|
return false
|
|
|
|
}
|
2015-09-12 15:10:03 +02:00
|
|
|
if application.Labels["traefik.enable"] == "false" {
|
2015-10-30 00:15:03 +01:00
|
|
|
log.Debugf("Filtering disabled marathon task %s", task.AppID)
|
2015-09-10 22:54:37 +02:00
|
|
|
return false
|
|
|
|
}
|
2015-10-30 00:15:03 +01:00
|
|
|
//filter healthchecks
|
|
|
|
if application.HasHealthChecks() {
|
|
|
|
if task.HasHealthCheckResults() {
|
|
|
|
for _, healthcheck := range task.HealthCheckResult {
|
|
|
|
// found one bad healthcheck, return false
|
|
|
|
if !healthcheck.Alive {
|
|
|
|
log.Debugf("Filtering marathon task %s with bad healthcheck", task.AppID)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Debugf("Filtering marathon task %s with bad healthcheck", task.AppID)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
2015-09-10 22:54:37 +02:00
|
|
|
return true
|
|
|
|
}, tasks.Tasks).([]marathon.Task)
|
|
|
|
|
|
|
|
//filter apps
|
|
|
|
filteredApps := fun.Filter(func(app marathon.Application) bool {
|
|
|
|
//get ports from app tasks
|
2015-09-12 15:10:03 +02:00
|
|
|
if !fun.Exists(func(task marathon.Task) bool {
|
|
|
|
if task.AppID == app.ID {
|
2015-09-10 22:54:37 +02:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
2015-09-12 15:10:03 +02:00
|
|
|
}, filteredTasks) {
|
2015-09-10 22:54:37 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}, applications.Apps).([]marathon.Application)
|
|
|
|
|
2015-09-09 22:39:08 +02:00
|
|
|
templateObjects := struct {
|
|
|
|
Applications []marathon.Application
|
|
|
|
Tasks []marathon.Task
|
|
|
|
Domain string
|
|
|
|
}{
|
2015-09-10 22:54:37 +02:00
|
|
|
filteredApps,
|
|
|
|
filteredTasks,
|
2015-09-09 22:39:08 +02:00
|
|
|
provider.Domain,
|
|
|
|
}
|
|
|
|
|
2015-09-21 18:05:56 +02:00
|
|
|
tmpl := template.New(provider.Filename).Funcs(MarathonFuncMap)
|
2015-09-12 15:10:03 +02:00
|
|
|
if len(provider.Filename) > 0 {
|
2015-09-11 19:25:49 +02:00
|
|
|
_, err := tmpl.ParseFiles(provider.Filename)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Error reading file", err)
|
|
|
|
return nil
|
|
|
|
}
|
2015-09-12 15:10:03 +02:00
|
|
|
} else {
|
2015-11-01 16:35:01 +01:00
|
|
|
buf, err := autogen.Asset("templates/marathon.tmpl")
|
2015-09-11 19:25:49 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Error("Error reading file", err)
|
|
|
|
}
|
|
|
|
_, err = tmpl.Parse(string(buf))
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Error reading file", err)
|
|
|
|
return nil
|
|
|
|
}
|
2015-09-09 22:39:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var buffer bytes.Buffer
|
|
|
|
|
|
|
|
err = tmpl.Execute(&buffer, templateObjects)
|
|
|
|
if err != nil {
|
2015-09-19 13:02:59 +02:00
|
|
|
log.Error("Error with marathon template:", err)
|
2015-09-09 22:39:08 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := toml.Decode(buffer.String(), configuration); err != nil {
|
2015-09-11 16:37:13 +02:00
|
|
|
log.Error("Error creating marathon configuration:", err)
|
2015-09-09 22:39:08 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return configuration
|
2015-09-10 22:54:37 +02:00
|
|
|
}
|
|
|
|
|
2015-10-27 00:26:35 +01:00
|
|
|
func getApplication(task marathon.Task, apps []marathon.Application) (marathon.Application, error) {
|
2015-09-10 22:54:37 +02:00
|
|
|
for _, application := range apps {
|
2015-09-12 15:10:03 +02:00
|
|
|
if application.ID == task.AppID {
|
2015-10-27 00:26:35 +01:00
|
|
|
return application, nil
|
2015-09-10 22:54:37 +02:00
|
|
|
}
|
|
|
|
}
|
2015-10-27 00:26:35 +01:00
|
|
|
return marathon.Application{}, errors.New("Application not found: " + task.AppID)
|
2015-09-12 15:10:03 +02:00
|
|
|
}
|
2015-10-23 09:49:19 +02:00
|
|
|
|
2015-11-02 19:48:34 +01:00
|
|
|
func (provider *Marathon) getLabel(application marathon.Application, label string) (string, error) {
|
2015-10-23 09:49:19 +02:00
|
|
|
for key, value := range application.Labels {
|
|
|
|
if key == label {
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "", errors.New("Label not found:" + label)
|
|
|
|
}
|
|
|
|
|
2015-11-02 19:48:34 +01:00
|
|
|
func (provider *Marathon) getEscapedName(name string) string {
|
2015-10-23 09:49:19 +02:00
|
|
|
return strings.Replace(name, "/", "", -1)
|
|
|
|
}
|
|
|
|
|
2015-11-01 19:29:47 +01:00
|
|
|
// GetFrontendValue returns the frontend value for the specified application, using
|
|
|
|
// it's label. It returns a default one if the label is not present.
|
2015-11-02 19:48:34 +01:00
|
|
|
func (provider *Marathon) GetFrontendValue(application marathon.Application) string {
|
2015-10-23 09:49:19 +02:00
|
|
|
if label, err := provider.getLabel(application, "traefik.frontend.value"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return provider.getEscapedName(application.ID) + "." + provider.Domain
|
|
|
|
}
|
|
|
|
|
2015-11-01 19:29:47 +01:00
|
|
|
// GetFrontendRule returns the frontend rule for the specified application, using
|
|
|
|
// it's label. It returns a default one (Host) if the label is not present.
|
2015-11-02 19:48:34 +01:00
|
|
|
func (provider *Marathon) GetFrontendRule(application marathon.Application) string {
|
2015-10-23 09:49:19 +02:00
|
|
|
if label, err := provider.getLabel(application, "traefik.frontend.rule"); err == nil {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return "Host"
|
|
|
|
}
|