2015-09-09 23:39:08 +03:00
package main
import (
"github.com/gambol99/go-marathon"
"log"
"github.com/leekchan/gtf"
"bytes"
"github.com/BurntSushi/toml"
"text/template"
"strings"
"strconv"
2015-09-10 23:54:37 +03:00
"github.com/BurntSushi/ty/fun"
2015-09-09 23:39:08 +03:00
)
type MarathonProvider struct {
2015-09-10 16:13:35 +03:00
Watch bool
Endpoint string
marathonClient marathon . Marathon
Domain string
Filename string
NetworkInterface string
2015-09-09 23:39:08 +03:00
}
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 {
2015-09-10 17:46:27 +03:00
if ( key == "traefik.host" ) {
2015-09-09 23:39:08 +03:00
return value
}
}
2015-09-10 01:22:01 +03:00
return strings . Replace ( application . ID , "/" , "" , 1 )
2015-09-09 23:39:08 +03:00
} ,
2015-09-10 17:14:08 +03:00
"getWeight" : func ( application marathon . Application ) string {
for key , value := range application . Labels {
2015-09-10 17:46:27 +03:00
if ( key == "traefik.weight" ) {
2015-09-10 17:14:08 +03:00
return value
}
}
return "0"
} ,
2015-09-09 23:39:08 +03:00
"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
2015-09-10 16:13:35 +03:00
config . EventsInterface = provider . NetworkInterface
2015-09-09 23:39:08 +03:00
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
2015-09-10 10:06:37 +03:00
update := make ( marathon . EventsChannel , 5 )
if ( provider . Watch ) {
if err := client . AddEventsListener ( update , marathon . EVENTS_APPLICATIONS ) ; err != nil {
log . Println ( "Failed to register for subscriptions, %s" , err )
} else {
go func ( ) {
for {
event := <- update
log . Println ( "Marathon event receveived" , event )
configuration := provider . loadMarathonConfig ( )
if ( configuration != nil ) {
configurationChan <- configuration
}
2015-09-10 00:09:16 +03:00
}
2015-09-10 10:06:37 +03:00
} ( )
}
2015-09-10 00:09:16 +03:00
}
2015-09-09 23:39:08 +03:00
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
}
2015-09-10 23:54:37 +03:00
2015-09-09 23:39:08 +03:00
tasks , err := provider . marathonClient . AllTasks ( )
if ( err != nil ) {
log . Println ( "Failed to create a client for marathon, error: %s" , err )
return nil
}
2015-09-10 23:54:37 +03:00
//filter tasks
filteredTasks := fun . Filter ( func ( task marathon . Task ) bool {
if ( len ( task . Ports ) == 0 ) {
log . Println ( "Filtering marathon task without port" , task . AppID )
return false
}
application := getApplication ( task , applications . Apps )
_ , err := strconv . Atoi ( application . Labels [ "traefik.port" ] )
if ( len ( application . Ports ) > 1 && err != nil ) {
log . Println ( "Filtering marathon task with more than 1 port and no traefik.port label" , task . AppID )
return false
}
if ( application . Labels [ "traefik.enable" ] == "false" ) {
log . Println ( "Filtering disabled marathon task" , task . AppID )
return false
}
return true
} , tasks . Tasks ) . ( [ ] marathon . Task )
//filter apps
filteredApps := fun . Filter ( func ( app marathon . Application ) bool {
//get ports from app tasks
if ( ! fun . Exists ( func ( task marathon . Task ) bool {
if ( task . AppID == app . ID ) {
return true
}
return false
} , filteredTasks ) ) {
return false
}
return true
} , applications . Apps ) . ( [ ] marathon . Application )
2015-09-09 23:39:08 +03:00
templateObjects := struct {
Applications [ ] marathon . Application
Tasks [ ] marathon . Task
Domain string
} {
2015-09-10 23:54:37 +03:00
filteredApps ,
filteredTasks ,
2015-09-09 23:39:08 +03:00
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
}
return configuration
2015-09-10 23:54:37 +03:00
}
func getApplication ( task marathon . Task , apps [ ] marathon . Application ) * marathon . Application {
for _ , application := range apps {
if ( application . ID == task . AppID ) {
return & application
}
}
return nil
2015-09-09 23:39:08 +03:00
}