2017-04-17 12:50:02 +02:00
package kubernetes
2016-02-08 21:57:32 +01:00
import (
2017-04-23 16:17:20 +02:00
"bufio"
"bytes"
"errors"
2017-07-03 10:06:32 +02:00
"flag"
2017-03-07 13:09:11 +01:00
"fmt"
"os"
2016-06-22 18:31:14 +02:00
"reflect"
2016-05-18 17:30:42 +01:00
"strconv"
2016-04-25 16:56:06 +02:00
"strings"
2016-02-08 21:57:32 +01:00
"text/template"
"time"
2016-09-12 21:06:21 +02:00
2016-12-30 09:21:13 +01:00
"github.com/cenk/backoff"
"github.com/containous/traefik/job"
2016-11-11 23:50:20 +01:00
"github.com/containous/traefik/log"
2017-04-17 12:50:02 +02:00
"github.com/containous/traefik/provider"
2016-11-11 23:50:20 +01:00
"github.com/containous/traefik/safe"
"github.com/containous/traefik/types"
2017-04-07 11:49:53 +01:00
"k8s.io/client-go/pkg/api/v1"
2017-04-23 16:17:20 +02:00
"k8s.io/client-go/pkg/apis/extensions/v1beta1"
2017-04-07 11:49:53 +01:00
"k8s.io/client-go/pkg/util/intstr"
2016-02-08 21:57:32 +01:00
)
2017-04-17 12:50:02 +02:00
var _ provider . Provider = ( * Provider ) ( nil )
2016-08-16 19:13:18 +02:00
2017-02-07 00:04:30 +01:00
const (
2017-07-10 16:58:12 +02:00
ruleTypePathPrefix = "PathPrefix"
ruleTypeReplacePath = "ReplacePath"
2017-04-30 11:22:07 +02:00
2017-11-28 06:36:03 -06:00
annotationKubernetesIngressClass = "kubernetes.io/ingress.class"
annotationKubernetesAuthRealm = "ingress.kubernetes.io/auth-realm"
annotationKubernetesAuthType = "ingress.kubernetes.io/auth-type"
annotationKubernetesAuthSecret = "ingress.kubernetes.io/auth-secret"
annotationKubernetesRewriteTarget = "ingress.kubernetes.io/rewrite-target"
annotationKubernetesWhitelistSourceRange = "ingress.kubernetes.io/whitelist-source-range"
annotationKubernetesSSLRedirect = "ingress.kubernetes.io/ssl-redirect"
annotationKubernetesHSTSMaxAge = "ingress.kubernetes.io/hsts-max-age"
annotationKubernetesHSTSIncludeSubdomains = "ingress.kubernetes.io/hsts-include-subdomains"
annotationKubernetesCustomRequestHeaders = "ingress.kubernetes.io/custom-request-headers"
annotationKubernetesCustomResponseHeaders = "ingress.kubernetes.io/custom-response-headers"
annotationKubernetesAllowedHosts = "ingress.kubernetes.io/allowed-hosts"
annotationKubernetesProxyHeaders = "ingress.kubernetes.io/proxy-headers"
annotationKubernetesSSLTemporaryRedirect = "ingress.kubernetes.io/ssl-temporary-redirect"
annotationKubernetesSSLHost = "ingress.kubernetes.io/ssl-host"
annotationKubernetesSSLProxyHeaders = "ingress.kubernetes.io/ssl-proxy-headers"
annotationKubernetesHSTSPreload = "ingress.kubernetes.io/hsts-preload"
annotationKubernetesForceHSTSHeader = "ingress.kubernetes.io/force-hsts"
annotationKubernetesFrameDeny = "ingress.kubernetes.io/frame-deny"
annotationKubernetesCustomFrameOptionsValue = "ingress.kubernetes.io/custom-frame-options-value"
annotationKubernetesContentTypeNosniff = "ingress.kubernetes.io/content-type-nosniff"
annotationKubernetesBrowserXSSFilter = "ingress.kubernetes.io/browser-xss-filter"
annotationKubernetesContentSecurityPolicy = "ingress.kubernetes.io/content-security-policy"
annotationKubernetesPublicKey = "ingress.kubernetes.io/public-key"
annotationKubernetesReferrerPolicy = "ingress.kubernetes.io/referrer-policy"
annotationKubernetesIsDevelopment = "ingress.kubernetes.io/is-development"
2017-02-07 00:04:30 +01:00
)
2017-04-23 16:17:20 +02:00
const traefikDefaultRealm = "traefik"
2017-04-17 12:50:02 +02:00
// Provider holds configurations of the provider.
type Provider struct {
2017-10-02 10:32:02 +02:00
provider . BaseProvider ` mapstructure:",squash" export:"true" `
2017-04-17 12:50:02 +02:00
Endpoint string ` description:"Kubernetes server endpoint (required for external cluster client)" `
Token string ` description:"Kubernetes bearer token (not needed for in-cluster client)" `
CertAuthFilePath string ` description:"Kubernetes certificate authority file path (not needed for in-cluster client)" `
2017-10-02 10:32:02 +02:00
DisablePassHostHeaders bool ` description:"Kubernetes disable PassHost Headers" export:"true" `
2017-11-20 02:12:03 +01:00
EnablePassTLSCert bool ` description:"Kubernetes enable Pass TLS Client Certs" export:"true" `
2017-10-02 10:32:02 +02:00
Namespaces Namespaces ` description:"Kubernetes namespaces" export:"true" `
LabelSelector string ` description:"Kubernetes api label selector to use" export:"true" `
2016-06-22 18:31:14 +02:00
lastConfiguration safe . Safe
2016-02-08 21:57:32 +01:00
}
2017-12-04 11:40:03 +01:00
func ( p * Provider ) newK8sClient ( ) ( Client , error ) {
2017-03-07 13:09:11 +01:00
withEndpoint := ""
2017-04-17 12:50:02 +02:00
if p . Endpoint != "" {
withEndpoint = fmt . Sprintf ( " with endpoint %v" , p . Endpoint )
2016-02-08 21:57:32 +01:00
}
2017-03-07 13:09:11 +01:00
if os . Getenv ( "KUBERNETES_SERVICE_HOST" ) != "" && os . Getenv ( "KUBERNETES_SERVICE_PORT" ) != "" {
2017-04-17 12:50:02 +02:00
log . Infof ( "Creating in-cluster Provider client%s\n" , withEndpoint )
return NewInClusterClient ( p . Endpoint )
2017-03-07 13:09:11 +01:00
}
2017-04-17 12:50:02 +02:00
log . Infof ( "Creating cluster-external Provider client%s\n" , withEndpoint )
return NewExternalClusterClient ( p . Endpoint , p . Token , p . CertAuthFilePath )
2016-04-19 19:23:08 +02:00
}
2017-04-17 12:50:02 +02:00
// Provide allows the k8s provider to provide configurations to traefik
2016-04-19 19:23:08 +02:00
// using the given configuration channel.
2017-04-17 12:50:02 +02:00
func ( p * Provider ) Provide ( configurationChan chan <- types . ConfigMessage , pool * safe . Pool , constraints types . Constraints ) error {
2017-07-03 10:06:32 +02:00
// Tell glog (used by client-go) to log into STDERR. Otherwise, we risk
// certain kinds of API errors getting logged into a directory not
// available in a `FROM scratch` Docker container, causing glog to abort
// hard with an exit code > 0.
flag . Set ( "logtostderr" , "true" )
2017-04-17 12:50:02 +02:00
k8sClient , err := p . newK8sClient ( )
2016-02-08 21:57:32 +01:00
if err != nil {
return err
}
2017-04-17 12:50:02 +02:00
p . Constraints = append ( p . Constraints , constraints ... )
2016-02-08 21:57:32 +01:00
pool . Go ( func ( stop chan bool ) {
operation := func ( ) error {
for {
2016-12-03 21:20:39 +01:00
stopWatch := make ( chan struct { } , 1 )
2016-05-19 20:09:01 +02:00
defer close ( stopWatch )
2017-04-17 12:50:02 +02:00
log . Debugf ( "Using label selector: '%s'" , p . LabelSelector )
2017-10-10 16:26:03 +02:00
eventsChan , err := k8sClient . WatchAll ( p . Namespaces , p . LabelSelector , stopWatch )
2016-04-25 16:56:06 +02:00
if err != nil {
log . Errorf ( "Error watching kubernetes events: %v" , err )
2016-05-19 20:09:01 +02:00
timer := time . NewTimer ( 1 * time . Second )
select {
case <- timer . C :
return err
case <- stop :
return nil
}
2016-04-25 16:56:06 +02:00
}
for {
select {
case <- stop :
return nil
case event := <- eventsChan :
2017-10-10 16:26:03 +02:00
log . Debugf ( "Received Kubernetes event kind %T" , event )
2017-04-17 12:50:02 +02:00
templateObjects , err := p . loadIngresses ( k8sClient )
2016-04-25 16:56:06 +02:00
if err != nil {
return err
}
2017-04-17 12:50:02 +02:00
if reflect . DeepEqual ( p . lastConfiguration . Get ( ) , templateObjects ) {
2017-10-10 16:26:03 +02:00
log . Debugf ( "Skipping Kubernetes event kind %T" , event )
2016-06-22 18:31:14 +02:00
} else {
2017-04-17 12:50:02 +02:00
p . lastConfiguration . Set ( templateObjects )
2016-06-22 18:31:14 +02:00
configurationChan <- types . ConfigMessage {
ProviderName : "kubernetes" ,
2017-04-17 12:50:02 +02:00
Configuration : p . loadConfig ( * templateObjects ) ,
2016-06-22 18:31:14 +02:00
}
2016-04-25 16:56:06 +02:00
}
2016-02-08 21:57:32 +01:00
}
}
}
}
notify := func ( err error , time time . Duration ) {
2017-06-28 01:32:19 +02:00
log . Errorf ( "Provider connection error: %s; retrying in %s" , err , time )
2016-02-08 21:57:32 +01:00
}
2016-12-08 13:32:12 +01:00
err := backoff . RetryNotify ( safe . OperationWithRecover ( operation ) , job . NewBackOff ( backoff . NewExponentialBackOff ( ) ) , notify )
2016-02-08 21:57:32 +01:00
if err != nil {
2017-06-28 01:32:19 +02:00
log . Errorf ( "Cannot connect to Provider: %s" , err )
2016-02-08 21:57:32 +01:00
}
} )
return nil
}
2017-04-17 12:50:02 +02:00
func ( p * Provider ) loadIngresses ( k8sClient Client ) ( * types . Configuration , error ) {
2017-10-10 16:26:03 +02:00
ingresses := k8sClient . GetIngresses ( )
2016-11-11 23:50:20 +01:00
2016-04-19 19:23:08 +02:00
templateObjects := types . Configuration {
2017-05-26 17:03:14 +02:00
Backends : map [ string ] * types . Backend { } ,
Frontends : map [ string ] * types . Frontend { } ,
2016-04-19 19:23:08 +02:00
}
for _ , i := range ingresses {
2017-05-26 17:03:14 +02:00
ingressClass := i . Annotations [ annotationKubernetesIngressClass ]
2017-03-03 11:30:22 -08:00
if ! shouldProcessIngress ( ingressClass ) {
continue
}
2016-04-19 19:23:08 +02:00
for _ , r := range i . Spec . Rules {
2016-12-08 13:32:52 +01:00
if r . HTTP == nil {
2017-05-26 17:03:14 +02:00
log . Warn ( "Error in ingress: HTTP is nil" )
2016-12-08 13:32:52 +01:00
continue
}
2017-07-07 15:27:54 -04:00
2016-04-19 19:23:08 +02:00
for _ , pa := range r . HTTP . Paths {
if _ , exists := templateObjects . Backends [ r . Host + pa . Path ] ; ! exists {
templateObjects . Backends [ r . Host + pa . Path ] = & types . Backend {
Servers : make ( map [ string ] types . Server ) ,
2017-01-25 08:11:00 -05:00
LoadBalancer : & types . LoadBalancer {
Method : "wrr" ,
} ,
2016-04-19 19:23:08 +02:00
}
}
2017-02-10 03:05:59 -08:00
2017-11-28 06:36:03 -06:00
passHostHeader := getBoolAnnotation ( i , types . LabelFrontendPassHostHeader , ! p . DisablePassHostHeaders )
passTLSCert := getBoolAnnotation ( i , types . LabelFrontendPassTLSCert , p . EnablePassTLSCert )
2017-05-26 17:03:14 +02:00
if realm := i . Annotations [ annotationKubernetesAuthRealm ] ; realm != "" && realm != traefikDefaultRealm {
2017-10-12 15:48:03 +02:00
log . Errorf ( "Value for annotation %q on ingress %s/%s invalid: no realm customization supported" , annotationKubernetesAuthRealm , i . ObjectMeta . Namespace , i . ObjectMeta . Name )
delete ( templateObjects . Backends , r . Host + pa . Path )
continue
2017-04-23 16:17:20 +02:00
}
2017-02-10 03:05:59 -08:00
2017-11-28 06:36:03 -06:00
entryPoints := getSliceAnnotation ( i , types . LabelFrontendEntryPoints )
2017-11-20 02:12:03 +01:00
2017-11-28 06:36:03 -06:00
whitelistSourceRange := getSliceAnnotation ( i , annotationKubernetesWhitelistSourceRange )
2017-04-30 11:22:07 +02:00
2016-04-19 19:23:08 +02:00
if _ , exists := templateObjects . Frontends [ r . Host + pa . Path ] ; ! exists {
2017-04-23 16:17:20 +02:00
basicAuthCreds , err := handleBasicAuthConfig ( i , k8sClient )
if err != nil {
2017-06-28 01:32:35 +02:00
log . Errorf ( "Failed to retrieve basic auth configuration for ingress %s/%s: %s" , i . ObjectMeta . Namespace , i . ObjectMeta . Name , err )
continue
2017-04-23 16:17:20 +02:00
}
2017-07-29 19:35:23 +03:00
2017-11-28 06:36:03 -06:00
priority := getPriority ( i )
headers := types . Headers {
CustomRequestHeaders : getMapAnnotation ( i , annotationKubernetesCustomRequestHeaders ) ,
CustomResponseHeaders : getMapAnnotation ( i , annotationKubernetesCustomResponseHeaders ) ,
AllowedHosts : getSliceAnnotation ( i , annotationKubernetesAllowedHosts ) ,
HostsProxyHeaders : getSliceAnnotation ( i , annotationKubernetesProxyHeaders ) ,
SSLRedirect : getBoolAnnotation ( i , annotationKubernetesSSLRedirect , false ) ,
SSLTemporaryRedirect : getBoolAnnotation ( i , annotationKubernetesSSLTemporaryRedirect , false ) ,
SSLHost : getStringAnnotation ( i , annotationKubernetesSSLHost ) ,
SSLProxyHeaders : getMapAnnotation ( i , annotationKubernetesSSLProxyHeaders ) ,
STSSeconds : getSTSSeconds ( i ) ,
STSIncludeSubdomains : getBoolAnnotation ( i , annotationKubernetesHSTSIncludeSubdomains , false ) ,
STSPreload : getBoolAnnotation ( i , annotationKubernetesHSTSPreload , false ) ,
ForceSTSHeader : getBoolAnnotation ( i , annotationKubernetesForceHSTSHeader , false ) ,
FrameDeny : getBoolAnnotation ( i , annotationKubernetesFrameDeny , false ) ,
CustomFrameOptionsValue : getStringAnnotation ( i , annotationKubernetesCustomFrameOptionsValue ) ,
ContentTypeNosniff : getBoolAnnotation ( i , annotationKubernetesContentTypeNosniff , false ) ,
BrowserXSSFilter : getBoolAnnotation ( i , annotationKubernetesBrowserXSSFilter , false ) ,
ContentSecurityPolicy : getStringAnnotation ( i , annotationKubernetesContentSecurityPolicy ) ,
PublicKey : getStringAnnotation ( i , annotationKubernetesPublicKey ) ,
ReferrerPolicy : getStringAnnotation ( i , annotationKubernetesReferrerPolicy ) ,
IsDevelopment : getBoolAnnotation ( i , annotationKubernetesIsDevelopment , false ) ,
}
2017-07-29 19:35:23 +03:00
2016-04-19 19:23:08 +02:00
templateObjects . Frontends [ r . Host + pa . Path ] = & types . Frontend {
2017-04-30 11:22:07 +02:00
Backend : r . Host + pa . Path ,
2017-11-20 02:12:03 +01:00
PassHostHeader : passHostHeader ,
PassTLSCert : passTLSCert ,
2017-04-30 11:22:07 +02:00
Routes : make ( map [ string ] types . Route ) ,
2017-07-29 19:35:23 +03:00
Priority : priority ,
2017-04-30 11:22:07 +02:00
BasicAuth : basicAuthCreds ,
WhitelistSourceRange : whitelistSourceRange ,
2017-12-15 11:48:03 +01:00
Redirect : getFrontendRedirect ( i ) ,
2017-11-20 02:12:03 +01:00
EntryPoints : entryPoints ,
2017-11-28 06:36:03 -06:00
Headers : headers ,
2016-04-19 19:23:08 +02:00
}
}
2016-05-25 13:16:19 +01:00
if len ( r . Host ) > 0 {
2017-01-20 14:16:05 +01:00
rule := "Host:" + r . Host
if strings . Contains ( r . Host , "*" ) {
rule = "HostRegexp:" + strings . Replace ( r . Host , "*" , "{subdomain:[A-Za-z0-9-_]+}" , 1 )
}
2016-05-25 13:16:19 +01:00
if _ , exists := templateObjects . Frontends [ r . Host + pa . Path ] . Routes [ r . Host ] ; ! exists {
templateObjects . Frontends [ r . Host + pa . Path ] . Routes [ r . Host ] = types . Route {
2017-01-20 14:16:05 +01:00
Rule : rule ,
2016-05-25 13:16:19 +01:00
}
2016-04-19 19:23:08 +02:00
}
}
2017-02-07 00:04:30 +01:00
2017-07-07 15:27:54 -04:00
rule := getRuleForPath ( pa , i )
if rule != "" {
2016-05-17 13:50:06 +03:00
templateObjects . Frontends [ r . Host + pa . Path ] . Routes [ pa . Path ] = types . Route {
2017-07-07 15:27:54 -04:00
Rule : rule ,
2016-04-19 19:23:08 +02:00
}
}
2017-02-07 00:04:30 +01:00
2016-11-11 23:50:20 +01:00
service , exists , err := k8sClient . GetService ( i . ObjectMeta . Namespace , pa . Backend . ServiceName )
2017-03-17 08:34:34 -07:00
if err != nil {
2017-03-22 18:56:14 +01:00
log . Errorf ( "Error while retrieving service information from k8s API %s/%s: %v" , i . ObjectMeta . Namespace , pa . Backend . ServiceName , err )
2017-03-17 08:34:34 -07:00
return nil , err
}
if ! exists {
2017-03-22 18:56:14 +01:00
log . Errorf ( "Service not found for %s/%s" , i . ObjectMeta . Namespace , pa . Backend . ServiceName )
2016-04-25 16:56:06 +02:00
delete ( templateObjects . Frontends , r . Host + pa . Path )
2016-05-26 00:53:51 +01:00
continue
2016-04-25 16:56:06 +02:00
}
2016-11-11 23:50:20 +01:00
2017-07-10 16:58:12 +02:00
if expression := service . Annotations [ types . LabelTraefikBackendCircuitbreaker ] ; expression != "" {
2017-02-03 11:47:48 -05:00
templateObjects . Backends [ r . Host + pa . Path ] . CircuitBreaker = & types . CircuitBreaker {
Expression : expression ,
}
}
2017-07-07 15:27:54 -04:00
2017-07-10 16:58:12 +02:00
if service . Annotations [ types . LabelBackendLoadbalancerMethod ] == "drr" {
2017-01-25 08:11:00 -05:00
templateObjects . Backends [ r . Host + pa . Path ] . LoadBalancer . Method = "drr"
}
2017-07-07 15:27:54 -04:00
2017-10-16 17:38:03 +02:00
if sticky := service . Annotations [ types . LabelBackendLoadbalancerSticky ] ; len ( sticky ) > 0 {
2017-10-12 17:50:03 +02:00
log . Warnf ( "Deprecated configuration found: %s. Please use %s." , types . LabelBackendLoadbalancerSticky , types . LabelBackendLoadbalancerStickiness )
2017-10-16 17:38:03 +02:00
templateObjects . Backends [ r . Host + pa . Path ] . LoadBalancer . Sticky = strings . EqualFold ( strings . TrimSpace ( sticky ) , "true" )
2017-10-10 11:10:02 +02:00
}
2017-10-16 17:38:03 +02:00
if service . Annotations [ types . LabelBackendLoadbalancerStickiness ] == "true" {
templateObjects . Backends [ r . Host + pa . Path ] . LoadBalancer . Stickiness = & types . Stickiness { }
if cookieName := service . Annotations [ types . LabelBackendLoadbalancerStickinessCookieName ] ; len ( cookieName ) > 0 {
templateObjects . Backends [ r . Host + pa . Path ] . LoadBalancer . Stickiness . CookieName = cookieName
2017-10-10 11:10:02 +02:00
}
2017-01-25 08:11:00 -05:00
}
2017-02-03 11:47:48 -05:00
2016-05-26 00:53:51 +01:00
protocol := "http"
for _ , port := range service . Spec . Ports {
if equalPorts ( port , pa . Backend . ServicePort ) {
if port . Port == 443 {
protocol = "https"
}
2017-07-07 15:27:54 -04:00
2017-02-09 17:25:38 -08:00
if service . Spec . Type == "ExternalName" {
url := protocol + "://" + service . Spec . ExternalName
name := url
templateObjects . Backends [ r . Host + pa . Path ] . Servers [ name ] = types . Server {
URL : url ,
2016-05-26 00:53:51 +01:00
Weight : 1 ,
2016-05-20 17:34:57 +01:00
}
2016-05-26 00:53:51 +01:00
} else {
2017-04-11 17:10:46 +02:00
endpoints , exists , err := k8sClient . GetEndpoints ( service . ObjectMeta . Namespace , service . ObjectMeta . Name )
if err != nil {
log . Errorf ( "Error retrieving endpoints %s/%s: %v" , service . ObjectMeta . Namespace , service . ObjectMeta . Name , err )
return nil , err
}
2017-02-14 11:53:35 -08:00
2017-04-11 17:10:46 +02:00
if ! exists {
2017-05-15 23:16:35 +02:00
log . Warnf ( "Endpoints not found for %s/%s" , service . ObjectMeta . Namespace , service . ObjectMeta . Name )
break
2017-03-14 15:59:13 +01:00
}
2017-04-11 17:10:46 +02:00
if len ( endpoints . Subsets ) == 0 {
2017-05-15 23:16:35 +02:00
log . Warnf ( "Endpoints not available for %s/%s" , service . ObjectMeta . Namespace , service . ObjectMeta . Name )
break
}
for _ , subset := range endpoints . Subsets {
for _ , address := range subset . Addresses {
url := protocol + "://" + address . IP + ":" + strconv . Itoa ( endpointPortNumber ( port , subset . Ports ) )
name := url
if address . TargetRef != nil && address . TargetRef . Name != "" {
name = address . TargetRef . Name
}
templateObjects . Backends [ r . Host + pa . Path ] . Servers [ name ] = types . Server {
URL : url ,
Weight : 1 ,
2016-05-20 17:34:57 +01:00
}
}
2016-04-20 13:26:51 +02:00
}
2016-04-19 19:23:08 +02:00
}
2016-05-26 00:53:51 +01:00
break
2016-04-19 19:23:08 +02:00
}
}
}
}
}
return & templateObjects , nil
}
2017-12-04 11:40:03 +01:00
func ( p * Provider ) loadConfig ( templateObjects types . Configuration ) * types . Configuration {
2017-11-28 06:36:03 -06:00
var FuncMap = template . FuncMap { }
configuration , err := p . GetConfiguration ( "templates/kubernetes.tmpl" , FuncMap , templateObjects )
if err != nil {
log . Error ( err )
2017-11-20 02:12:03 +01:00
}
2017-11-28 06:36:03 -06:00
return configuration
2017-11-20 02:12:03 +01:00
}
2017-11-28 06:36:03 -06:00
func getSTSSeconds ( i * v1beta1 . Ingress ) int64 {
value , err := strconv . ParseInt ( i . ObjectMeta . Annotations [ annotationKubernetesHSTSMaxAge ] , 10 , 64 )
if err == nil && value > 0 {
return value
2017-11-20 02:12:03 +01:00
}
2017-11-28 06:36:03 -06:00
return 0
2017-11-20 02:12:03 +01:00
}
2017-07-07 15:27:54 -04:00
func getRuleForPath ( pa v1beta1 . HTTPIngressPath , i * v1beta1 . Ingress ) string {
if len ( pa . Path ) == 0 {
return ""
}
2017-07-10 16:58:12 +02:00
ruleType := i . Annotations [ types . LabelFrontendRuleType ]
2017-07-07 15:27:54 -04:00
if ruleType == "" {
ruleType = ruleTypePathPrefix
}
2017-11-27 18:22:03 +08:00
rules := [ ] string { ruleType + ":" + pa . Path }
2017-07-07 15:27:54 -04:00
if rewriteTarget := i . Annotations [ annotationKubernetesRewriteTarget ] ; rewriteTarget != "" {
2017-11-27 18:22:03 +08:00
rules = append ( rules , ruleTypeReplacePath + ":" + rewriteTarget )
2017-07-07 15:27:54 -04:00
}
2017-11-27 18:22:03 +08:00
return strings . Join ( rules , ";" )
2017-07-07 15:27:54 -04:00
}
2017-11-28 06:36:03 -06:00
func getPriority ( i * v1beta1 . Ingress ) int {
2017-08-18 16:14:03 +02:00
priority := 0
2017-07-29 19:35:23 +03:00
priorityRaw , ok := i . Annotations [ types . LabelFrontendPriority ]
if ok {
priorityParsed , err := strconv . Atoi ( priorityRaw )
if err == nil {
priority = priorityParsed
} else {
log . Errorf ( "Error in ingress: failed to parse %q value %q." , types . LabelFrontendPriority , priorityRaw )
}
}
return priority
}
2017-04-23 16:17:20 +02:00
func handleBasicAuthConfig ( i * v1beta1 . Ingress , k8sClient Client ) ( [ ] string , error ) {
2017-05-26 17:03:14 +02:00
authType , exists := i . Annotations [ annotationKubernetesAuthType ]
2017-04-23 16:17:20 +02:00
if ! exists {
return nil , nil
}
if strings . ToLower ( authType ) != "basic" {
2017-06-28 01:32:19 +02:00
return nil , fmt . Errorf ( "unsupported auth-type on annotation ingress.kubernetes.io/auth-type: %q" , authType )
2017-04-23 16:17:20 +02:00
}
2017-05-26 17:03:14 +02:00
authSecret := i . Annotations [ annotationKubernetesAuthSecret ]
2017-04-23 16:17:20 +02:00
if authSecret == "" {
2017-06-28 01:32:19 +02:00
return nil , errors . New ( "auth-secret annotation ingress.kubernetes.io/auth-secret must be set" )
2017-04-23 16:17:20 +02:00
}
basicAuthCreds , err := loadAuthCredentials ( i . Namespace , authSecret , k8sClient )
if err != nil {
2017-06-28 01:32:19 +02:00
return nil , fmt . Errorf ( "failed to load auth credentials: %s" , err )
2017-04-23 16:17:20 +02:00
}
return basicAuthCreds , nil
}
func loadAuthCredentials ( namespace , secretName string , k8sClient Client ) ( [ ] string , error ) {
secret , ok , err := k8sClient . GetSecret ( namespace , secretName )
switch { // keep order of case conditions
case err != nil :
return nil , fmt . Errorf ( "failed to fetch secret %q/%q: %s" , namespace , secretName , err )
case ! ok :
return nil , fmt . Errorf ( "secret %q/%q not found" , namespace , secretName )
case secret == nil :
2017-06-28 01:32:19 +02:00
return nil , fmt . Errorf ( "data for secret %q/%q must not be nil" , namespace , secretName )
2017-04-23 16:17:20 +02:00
case len ( secret . Data ) != 1 :
2017-06-28 01:32:19 +02:00
return nil , fmt . Errorf ( "found %d elements for secret %q/%q, must be single element exactly" , len ( secret . Data ) , namespace , secretName )
2017-04-23 16:17:20 +02:00
default :
}
var firstSecret [ ] byte
for _ , v := range secret . Data {
firstSecret = v
break
}
creds := make ( [ ] string , 0 )
scanner := bufio . NewScanner ( bytes . NewReader ( firstSecret ) )
for scanner . Scan ( ) {
if cred := scanner . Text ( ) ; cred != "" {
creds = append ( creds , cred )
}
}
2017-06-28 01:32:19 +02:00
if len ( creds ) == 0 {
return nil , fmt . Errorf ( "secret %q/%q does not contain any credentials" , namespace , secretName )
}
2017-04-23 16:17:20 +02:00
return creds , nil
}
2016-11-11 23:50:20 +01:00
func endpointPortNumber ( servicePort v1 . ServicePort , endpointPorts [ ] v1 . EndpointPort ) int {
2016-05-20 17:34:57 +01:00
if len ( endpointPorts ) > 0 {
//name is optional if there is only one port
port := endpointPorts [ 0 ]
for _ , endpointPort := range endpointPorts {
if servicePort . Name == endpointPort . Name {
port = endpointPort
}
}
return int ( port . Port )
}
2016-11-11 23:50:20 +01:00
return int ( servicePort . Port )
2016-05-20 17:34:57 +01:00
}
2016-11-11 23:50:20 +01:00
func equalPorts ( servicePort v1 . ServicePort , ingressPort intstr . IntOrString ) bool {
if int ( servicePort . Port ) == ingressPort . IntValue ( ) {
2016-05-18 17:30:42 +01:00
return true
}
if servicePort . Name != "" && servicePort . Name == ingressPort . String ( ) {
return true
}
return false
}
2017-03-03 11:30:22 -08:00
func shouldProcessIngress ( ingressClass string ) bool {
switch ingressClass {
case "" , "traefik" :
return true
default :
return false
}
}
2017-12-15 11:48:03 +01:00
// TODO will be rewrite when merge on master
func getFrontendRedirect ( i * v1beta1 . Ingress ) * types . Redirect {
frontendRedirectEntryPoint , ok := i . Annotations [ types . LabelFrontendRedirectEntryPoint ]
frep := ok && len ( frontendRedirectEntryPoint ) > 0
frontendRedirectRegex , ok := i . Annotations [ types . LabelFrontendRedirectRegex ]
frrg := ok && len ( frontendRedirectRegex ) > 0
frontendRedirectReplacement , ok := i . Annotations [ types . LabelFrontendRedirectReplacement ]
frrp := ok && len ( frontendRedirectReplacement ) > 0
if frep || frrg && frrp {
return & types . Redirect {
EntryPoint : frontendRedirectEntryPoint ,
Regex : frontendRedirectRegex ,
Replacement : frontendRedirectReplacement ,
}
}
return nil
}