2018-07-04 00:42:03 +03:00
package integration
import (
"fmt"
"net/http"
"os"
"time"
"github.com/gambol99/go-marathon"
"github.com/go-check/check"
2020-09-16 16:46:04 +03:00
"github.com/traefik/traefik/v2/integration/try"
2018-07-04 00:42:03 +03:00
checker "github.com/vdemeester/shakers"
)
2020-05-11 13:06:07 +03:00
// Marathon test suites (using libcompose).
2018-07-04 00:42:03 +03:00
type MarathonSuite15 struct {
BaseSuite
marathonURL string
}
func ( s * MarathonSuite15 ) SetUpSuite ( c * check . C ) {
s . createComposeProject ( c , "marathon15" )
s . composeProject . Start ( c )
marathonIPAddr := s . composeProject . Container ( c , containerNameMarathon ) . NetworkSettings . IPAddress
c . Assert ( marathonIPAddr , checker . Not ( checker . HasLen ) , 0 )
s . marathonURL = "http://" + marathonIPAddr + ":8080"
// Wait for Marathon readiness prior to creating the client so that we
// don't run into the "all cluster members down" state right from the
// start.
err := try . GetRequest ( s . marathonURL + "/v2/leader" , 1 * time . Minute , try . StatusCodeIs ( http . StatusOK ) )
c . Assert ( err , checker . IsNil )
// Add entry for Mesos slave container IP address in the hosts file so
// that Traefik can properly forward traffic.
// This is necessary as long as we are still using the docker-compose v1
// spec. Once we switch to v2 or higher, we can have both the test/builder
// container and the Mesos slave container join the same custom network and
// enjoy DNS-discoverable container host names.
mesosSlaveIPAddr := s . composeProject . Container ( c , containerNameMesosSlave ) . NetworkSettings . IPAddress
c . Assert ( mesosSlaveIPAddr , checker . Not ( checker . HasLen ) , 0 )
err = s . extendDockerHostsFile ( containerNameMesosSlave , mesosSlaveIPAddr )
c . Assert ( err , checker . IsNil )
}
// extendDockerHostsFile extends the hosts file (/etc/hosts) by the given
// host/IP address mapping if we are running inside a container.
func ( s * MarathonSuite15 ) extendDockerHostsFile ( host , ipAddr string ) error {
const hostsFile = "/etc/hosts"
// Determine if the run inside a container. The most reliable way to
// do this is to inject an indicator, which we do in terms of an
// environment variable.
// (See also https://groups.google.com/d/topic/docker-user/JOGE7AnJ3Gw/discussion.)
if os . Getenv ( "CONTAINER" ) == "DOCKER" {
// We are running inside a container -- extend the hosts file.
2020-07-07 15:42:03 +03:00
file , err := os . OpenFile ( hostsFile , os . O_APPEND | os . O_WRONLY , 0 o600 )
2018-07-04 00:42:03 +03:00
if err != nil {
return err
}
defer file . Close ( )
if _ , err = file . WriteString ( fmt . Sprintf ( "%s\t%s\n" , ipAddr , host ) ) ; err != nil {
return err
}
}
return nil
}
func ( s * MarathonSuite15 ) TestConfigurationUpdate ( c * check . C ) {
// Start Traefik.
file := s . adaptFile ( c , "fixtures/marathon/simple.toml" , struct {
MarathonURL string
} { s . marathonURL } )
defer os . Remove ( file )
cmd , display := s . traefikCmd ( withConfigFile ( file ) )
defer display ( c )
err := cmd . Start ( )
c . Assert ( err , checker . IsNil )
2020-10-09 10:32:03 +03:00
defer s . killCmd ( cmd )
2018-07-04 00:42:03 +03:00
// Wait for Traefik to turn ready.
err = try . GetRequest ( "http://127.0.0.1:8000/" , 2 * time . Second , try . StatusCodeIs ( http . StatusNotFound ) )
c . Assert ( err , checker . IsNil )
// Prepare Marathon client.
config := marathon . NewDefaultConfig ( )
config . URL = s . marathonURL
client , err := marathon . NewClient ( config )
c . Assert ( err , checker . IsNil )
// Create test application to be deployed.
app := marathon . NewDockerApplication ( ) .
Name ( "/whoami" ) .
CPU ( 0.1 ) .
Memory ( 32 ) .
EmptyNetworks ( ) .
2019-03-14 11:30:04 +03:00
AddLabel ( "traefik.http.Routers.rt.Rule" , "PathPrefix(`/service`)" )
2018-07-04 00:42:03 +03:00
app . Container .
Expose ( 80 ) .
Docker .
2020-09-16 16:46:04 +03:00
Container ( "traefik/whoami" )
2018-07-04 00:42:03 +03:00
* app . Networks = append ( * app . Networks , * marathon . NewBridgePodNetwork ( ) )
// Deploy the test application.
deployApplication ( c , client , app )
// Query application via Traefik.
err = try . GetRequest ( "http://127.0.0.1:8000/service" , 30 * time . Second , try . StatusCodeIs ( http . StatusOK ) )
c . Assert ( err , checker . IsNil )
// Create test application with services to be deployed.
app = marathon . NewDockerApplication ( ) .
Name ( "/whoami" ) .
CPU ( 0.1 ) .
Memory ( 32 ) .
EmptyNetworks ( ) .
2019-03-14 11:30:04 +03:00
AddLabel ( "traefik.http.Routers.app.Rule" , "PathPrefix(`/app`)" )
2018-07-04 00:42:03 +03:00
app . Container .
Expose ( 80 ) .
Docker .
2020-09-16 16:46:04 +03:00
Container ( "traefik/whoami" )
2018-07-04 00:42:03 +03:00
* app . Networks = append ( * app . Networks , * marathon . NewBridgePodNetwork ( ) )
// Deploy the test application.
deployApplication ( c , client , app )
// Query application via Traefik.
err = try . GetRequest ( "http://127.0.0.1:8000/app" , 30 * time . Second , try . StatusCodeIs ( http . StatusOK ) )
c . Assert ( err , checker . IsNil )
}