Marathon provider
This commit is contained in:
parent
4a96f19221
commit
9a840d1904
5 changed files with 156 additions and 27 deletions
22
docker.go
22
docker.go
|
@ -8,6 +8,15 @@ import (
|
||||||
"text/template"
|
"text/template"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type DockerProvider struct {
|
||||||
|
Watch bool
|
||||||
|
Endpoint string
|
||||||
|
dockerClient *docker.Client
|
||||||
|
Filename string
|
||||||
|
Domain string
|
||||||
|
}
|
||||||
|
|
||||||
var DockerFuncMap = template.FuncMap{
|
var DockerFuncMap = template.FuncMap{
|
||||||
"getBackend": func(container docker.Container) string {
|
"getBackend": func(container docker.Container) string {
|
||||||
for key, value := range container.Config.Labels {
|
for key, value := range container.Config.Labels {
|
||||||
|
@ -30,16 +39,12 @@ var DockerFuncMap = template.FuncMap{
|
||||||
},
|
},
|
||||||
"getHost": getHost,
|
"getHost": getHost,
|
||||||
}
|
}
|
||||||
type DockerProvider struct {
|
|
||||||
Watch bool
|
|
||||||
Endpoint string
|
|
||||||
dockerClient *docker.Client
|
|
||||||
Filename string
|
|
||||||
Domain string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (provider *DockerProvider) Provide(configurationChan chan <- *Configuration) {
|
func (provider *DockerProvider) Provide(configurationChan chan <- *Configuration) {
|
||||||
provider.dockerClient, _ = docker.NewClient(provider.Endpoint)
|
if client, err := docker.NewClient(provider.Endpoint); err != nil {
|
||||||
|
log.Fatalf("Failed to create a client for docker, error: %s", err)
|
||||||
|
} else {
|
||||||
|
provider.dockerClient = client
|
||||||
dockerEvents := make(chan *docker.APIEvents)
|
dockerEvents := make(chan *docker.APIEvents)
|
||||||
if (provider.Watch) {
|
if (provider.Watch) {
|
||||||
provider.dockerClient.AddEventListener(dockerEvents)
|
provider.dockerClient.AddEventListener(dockerEvents)
|
||||||
|
@ -58,6 +63,7 @@ func (provider *DockerProvider) Provide(configurationChan chan <- *Configuration
|
||||||
configuration := provider.loadDockerConfig()
|
configuration := provider.loadDockerConfig()
|
||||||
configurationChan <- configuration
|
configurationChan <- configuration
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (provider *DockerProvider) loadDockerConfig() *Configuration {
|
func (provider *DockerProvider) loadDockerConfig() *Configuration {
|
||||||
configuration := new(Configuration)
|
configuration := new(Configuration)
|
||||||
|
|
100
marathon.go
Normal file
100
marathon.go
Normal file
|
@ -0,0 +1,100 @@
|
||||||
|
package main
|
||||||
|
import (
|
||||||
|
"github.com/gambol99/go-marathon"
|
||||||
|
"log"
|
||||||
|
"github.com/leekchan/gtf"
|
||||||
|
"bytes"
|
||||||
|
"github.com/BurntSushi/toml"
|
||||||
|
"text/template"
|
||||||
|
"strings"
|
||||||
|
"strconv"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MarathonProvider struct {
|
||||||
|
Watch bool
|
||||||
|
Endpoint string
|
||||||
|
marathonClient marathon.Marathon
|
||||||
|
Domain string
|
||||||
|
Filename string
|
||||||
|
}
|
||||||
|
|
||||||
|
var MarathonFuncMap = template.FuncMap{
|
||||||
|
"getPort": func(task marathon.Task) string {
|
||||||
|
for _, port := range task.Ports {
|
||||||
|
return strconv.Itoa(port)
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
},
|
||||||
|
"getHost": func(application marathon.Application) string {
|
||||||
|
for key, value := range application.Labels {
|
||||||
|
if (key == "træfik.host") {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return strings.TrimPrefix(application.ID, "/")
|
||||||
|
},
|
||||||
|
"replace": func(s1 string, s2 string, s3 string) string {
|
||||||
|
return strings.Replace(s3, s1, s2, -1)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
func (provider *MarathonProvider) Provide(configurationChan chan <- *Configuration) {
|
||||||
|
config := marathon.NewDefaultConfig()
|
||||||
|
config.URL = provider.Endpoint
|
||||||
|
if client, err := marathon.NewClient(config); err != nil {
|
||||||
|
log.Println("Failed to create a client for marathon, error: %s", err)
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
provider.marathonClient = client
|
||||||
|
configuration := provider.loadMarathonConfig()
|
||||||
|
configurationChan <- configuration
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (provider *MarathonProvider) loadMarathonConfig() *Configuration {
|
||||||
|
configuration := new(Configuration)
|
||||||
|
|
||||||
|
applications, err := provider.marathonClient.Applications(nil)
|
||||||
|
if (err != nil) {
|
||||||
|
log.Println("Failed to create a client for marathon, error: %s", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
tasks, err := provider.marathonClient.AllTasks()
|
||||||
|
if (err != nil) {
|
||||||
|
log.Println("Failed to create a client for marathon, error: %s", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
templateObjects := struct {
|
||||||
|
Applications []marathon.Application
|
||||||
|
Tasks []marathon.Task
|
||||||
|
Domain string
|
||||||
|
}{
|
||||||
|
applications.Apps,
|
||||||
|
tasks.Tasks,
|
||||||
|
provider.Domain,
|
||||||
|
}
|
||||||
|
|
||||||
|
gtf.Inject(MarathonFuncMap)
|
||||||
|
tmpl, err := template.New(provider.Filename).Funcs(MarathonFuncMap).ParseFiles(provider.Filename)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Error reading file:", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var buffer bytes.Buffer
|
||||||
|
|
||||||
|
err = tmpl.Execute(&buffer, templateObjects)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Error with docker template:", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := toml.Decode(buffer.String(), configuration); err != nil {
|
||||||
|
log.Println("Error creating marathon configuration:", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Println(buffer.String())
|
||||||
|
|
||||||
|
return configuration
|
||||||
|
}
|
12
marathon.tmpl
Normal file
12
marathon.tmpl
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
[backends]{{range .Tasks}}
|
||||||
|
[backends.backend{{.AppID | replace "/" "-"}}.servers.server-{{.ID | replace "." "-"}}]
|
||||||
|
url = "http://{{.Host}}:{{getPort .}}"
|
||||||
|
{{end}}
|
||||||
|
|
||||||
|
[routes]{{range .Applications}}
|
||||||
|
[routes.route{{.ID | replace "/" "-"}}]
|
||||||
|
backend = ".backend{{.ID | replace "/" "-"}}"
|
||||||
|
[routes.route-{{getHost .}}.rules.rule-host-{{getHost .}}]
|
||||||
|
category = "Host"
|
||||||
|
value = "{{getHost .}}.{{$.Domain}}"
|
||||||
|
{{end}}
|
|
@ -23,6 +23,7 @@ type FileConfiguration struct {
|
||||||
Docker *DockerProvider
|
Docker *DockerProvider
|
||||||
File *FileProvider
|
File *FileProvider
|
||||||
Web *WebProvider
|
Web *WebProvider
|
||||||
|
Marathon *MarathonProvider
|
||||||
}
|
}
|
||||||
|
|
||||||
var srv *graceful.Server
|
var srv *graceful.Server
|
||||||
|
@ -58,6 +59,10 @@ func main() {
|
||||||
providers = append(providers, configuration.Docker)
|
providers = append(providers, configuration.Docker)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (configuration.Marathon != nil) {
|
||||||
|
providers = append(providers, configuration.Marathon)
|
||||||
|
}
|
||||||
|
|
||||||
if (configuration.File != nil) {
|
if (configuration.File != nil) {
|
||||||
if (len(configuration.File.Filename) == 0) {
|
if (len(configuration.File.Filename) == 0) {
|
||||||
// no filename, setting to global config file
|
// no filename, setting to global config file
|
||||||
|
|
14
træfik.toml
14
træfik.toml
|
@ -1,16 +1,22 @@
|
||||||
port = ":8001"
|
port = ":8001"
|
||||||
|
|
||||||
[docker]
|
#[docker]
|
||||||
endpoint = "unix:///var/run/docker.sock"
|
#endpoint = "unix:///var/run/docker.sock"
|
||||||
|
#watch = true
|
||||||
|
#domain = "toto.fr"
|
||||||
|
#filename = "docker.tmpl"
|
||||||
|
|
||||||
|
[marathon]
|
||||||
|
endpoint = "http://127.0.0.1:8080"
|
||||||
watch = true
|
watch = true
|
||||||
domain = "toto.fr"
|
domain = "toto.fr"
|
||||||
filename = "docker.tmpl"
|
filename = "marathon.tmpl"
|
||||||
|
|
||||||
[web]
|
[web]
|
||||||
address = ":8010"
|
address = ":8010"
|
||||||
|
|
||||||
#[file]
|
#[file]
|
||||||
#filename = "træfik.toml"
|
#filename = "rules.toml"
|
||||||
#watch = true
|
#watch = true
|
||||||
|
|
||||||
[backends]
|
[backends]
|
||||||
|
|
Loading…
Add table
Reference in a new issue