2017-12-02 19:26:44 +01:00
|
|
|
package docker
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/containous/traefik/provider"
|
|
|
|
"github.com/containous/traefik/provider/label"
|
2017-12-18 16:26:04 +01:00
|
|
|
"github.com/containous/traefik/types"
|
2017-12-02 19:26:44 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// Specific functions
|
|
|
|
|
|
|
|
// Extract rule from labels for a given service and a given docker container
|
|
|
|
func (p Provider) getServiceFrontendRule(container dockerData, serviceName string) string {
|
|
|
|
if value, ok := getServiceLabels(container, serviceName)[label.SuffixFrontendRule]; ok {
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
return p.getFrontendRule(container)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if for the given container, we find labels that are defining services
|
|
|
|
func hasServices(container dockerData) bool {
|
|
|
|
return len(label.ExtractServiceProperties(container.Labels)) > 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gets array of service names for a given container
|
|
|
|
func getServiceNames(container dockerData) []string {
|
|
|
|
labelServiceProperties := label.ExtractServiceProperties(container.Labels)
|
|
|
|
keys := make([]string, 0, len(labelServiceProperties))
|
|
|
|
for k := range labelServiceProperties {
|
|
|
|
keys = append(keys, k)
|
|
|
|
}
|
|
|
|
return keys
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkServiceLabelPort checks if all service names have a port service label
|
|
|
|
// or if port container label exists for default value
|
|
|
|
func checkServiceLabelPort(container dockerData) error {
|
|
|
|
// If port container label is present, there is a default values for all ports, use it for the check
|
|
|
|
_, err := strconv.Atoi(container.Labels[label.TraefikPort])
|
|
|
|
if err != nil {
|
|
|
|
serviceLabelPorts := make(map[string]struct{})
|
|
|
|
serviceLabels := make(map[string]struct{})
|
|
|
|
for lbl := range container.Labels {
|
|
|
|
// Get all port service labels
|
|
|
|
portLabel := label.PortRegexp.FindStringSubmatch(lbl)
|
|
|
|
if len(portLabel) > 0 {
|
|
|
|
serviceLabelPorts[portLabel[0]] = struct{}{}
|
|
|
|
}
|
|
|
|
// Get only one instance of all service names from service labels
|
|
|
|
servicesLabelNames := label.ServicesPropertiesRegexp.FindStringSubmatch(lbl)
|
2017-12-18 16:26:04 +01:00
|
|
|
|
|
|
|
if len(servicesLabelNames) > 0 && !strings.HasPrefix(lbl, label.TraefikFrontend) {
|
2017-12-02 19:26:44 +01:00
|
|
|
serviceLabels[strings.Split(servicesLabelNames[0], ".")[1]] = struct{}{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If the number of service labels is different than the number of port services label
|
|
|
|
// there is an error
|
|
|
|
if len(serviceLabels) == len(serviceLabelPorts) {
|
|
|
|
for labelPort := range serviceLabelPorts {
|
|
|
|
_, err = strconv.Atoi(container.Labels[labelPort])
|
|
|
|
if err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = errors.New("port service labels missing, please use traefik.port as default value or define all port service labels")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Extract backend from labels for a given service and a given docker container
|
|
|
|
func getServiceBackend(container dockerData, serviceName string) string {
|
|
|
|
if value, ok := getServiceLabels(container, serviceName)[label.SuffixFrontendBackend]; ok {
|
2018-01-02 14:49:11 +01:00
|
|
|
return provider.Normalize(container.ServiceName + "-" + value)
|
2017-12-02 19:26:44 +01:00
|
|
|
}
|
2018-01-02 14:49:11 +01:00
|
|
|
return provider.Normalize(container.ServiceName + "-" + getBackend(container) + "-" + serviceName)
|
2017-12-02 19:26:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Extract port from labels for a given service and a given docker container
|
|
|
|
func getServicePort(container dockerData, serviceName string) string {
|
|
|
|
if value, ok := getServiceLabels(container, serviceName)[label.SuffixPort]; ok {
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
return getPort(container)
|
|
|
|
}
|
|
|
|
|
2017-12-15 22:16:48 +01:00
|
|
|
func hasServiceRedirect(container dockerData, serviceName string) bool {
|
|
|
|
serviceLabels := getServiceLabels(container, serviceName)
|
|
|
|
if len(serviceLabels) == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return label.Has(serviceLabels, label.SuffixFrontendRedirectEntryPoint) ||
|
|
|
|
label.Has(serviceLabels, label.SuffixFrontendRedirectRegex) && label.Has(serviceLabels, label.SuffixFrontendRedirectReplacement)
|
|
|
|
}
|
|
|
|
|
2017-12-18 16:26:04 +01:00
|
|
|
func hasServiceErrorPages(container dockerData, serviceName string) bool {
|
|
|
|
serviceLabels := getServiceLabels(container, serviceName)
|
|
|
|
return label.HasPrefix(serviceLabels, label.BaseFrontendErrorPage)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getServiceErrorPages(container dockerData, serviceName string) map[string]*types.ErrorPage {
|
|
|
|
serviceLabels := getServiceLabels(container, serviceName)
|
|
|
|
return label.ParseErrorPages(serviceLabels, label.BaseFrontendErrorPage, label.RegexpBaseFrontendErrorPage)
|
|
|
|
}
|
|
|
|
|
2017-12-18 18:06:12 +01:00
|
|
|
func getServiceRateLimits(container dockerData, serviceName string) map[string]*types.Rate {
|
|
|
|
serviceLabels := getServiceLabels(container, serviceName)
|
|
|
|
return label.ParseRateSets(serviceLabels, label.BaseFrontendRateLimit, label.RegexpBaseFrontendRateLimit)
|
|
|
|
}
|
|
|
|
|
2017-12-02 19:26:44 +01:00
|
|
|
// Service label functions
|
|
|
|
|
2017-12-07 05:26:03 +08:00
|
|
|
func getFuncServiceMapLabel(labelSuffix string) func(container dockerData, serviceName string) map[string]string {
|
|
|
|
return func(container dockerData, serviceName string) map[string]string {
|
|
|
|
return getServiceMapLabel(container, serviceName, labelSuffix)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-02 19:26:44 +01:00
|
|
|
func getFuncServiceSliceStringLabel(labelSuffix string) func(container dockerData, serviceName string) []string {
|
|
|
|
return func(container dockerData, serviceName string) []string {
|
|
|
|
return getServiceSliceStringLabel(container, serviceName, labelSuffix)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func getFuncServiceStringLabel(labelSuffix string, defaultValue string) func(container dockerData, serviceName string) string {
|
|
|
|
return func(container dockerData, serviceName string) string {
|
|
|
|
return getServiceStringLabel(container, serviceName, labelSuffix, defaultValue)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-16 12:42:32 +01:00
|
|
|
func getFuncServiceBoolLabel(labelSuffix string, defaultValue bool) func(container dockerData, serviceName string) bool {
|
|
|
|
return func(container dockerData, serviceName string) bool {
|
|
|
|
return getServiceBoolLabel(container, serviceName, labelSuffix, defaultValue)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-16 13:24:07 +01:00
|
|
|
func getFuncServiceIntLabel(labelSuffix string, defaultValue int) func(container dockerData, serviceName string) int {
|
|
|
|
return func(container dockerData, serviceName string) int {
|
|
|
|
return getServiceIntLabel(container, serviceName, labelSuffix, defaultValue)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-02 19:26:44 +01:00
|
|
|
func hasFuncServiceLabel(labelSuffix string) func(container dockerData, serviceName string) bool {
|
|
|
|
return func(container dockerData, serviceName string) bool {
|
|
|
|
return hasServiceLabel(container, serviceName, labelSuffix)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func hasServiceLabel(container dockerData, serviceName string, labelSuffix string) bool {
|
|
|
|
value, ok := getServiceLabels(container, serviceName)[labelSuffix]
|
|
|
|
if ok && len(value) > 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return label.Has(container.Labels, label.Prefix+labelSuffix)
|
|
|
|
}
|
|
|
|
|
2017-12-07 05:26:03 +08:00
|
|
|
func getServiceMapLabel(container dockerData, serviceName string, labelSuffix string) map[string]string {
|
|
|
|
if value, ok := getServiceLabels(container, serviceName)[labelSuffix]; ok {
|
|
|
|
lblName := label.GetServiceLabel(labelSuffix, serviceName)
|
|
|
|
return label.ParseMapValue(lblName, value)
|
|
|
|
}
|
|
|
|
return label.GetMapValue(container.Labels, label.Prefix+labelSuffix)
|
|
|
|
}
|
|
|
|
|
2017-12-02 19:26:44 +01:00
|
|
|
func getServiceSliceStringLabel(container dockerData, serviceName string, labelSuffix string) []string {
|
|
|
|
if value, ok := getServiceLabels(container, serviceName)[labelSuffix]; ok {
|
|
|
|
return label.SplitAndTrimString(value, ",")
|
|
|
|
}
|
|
|
|
return label.GetSliceStringValue(container.Labels, label.Prefix+labelSuffix)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getServiceStringLabel(container dockerData, serviceName string, labelSuffix string, defaultValue string) string {
|
|
|
|
if value, ok := getServiceLabels(container, serviceName)[labelSuffix]; ok {
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
return label.GetStringValue(container.Labels, label.Prefix+labelSuffix, defaultValue)
|
|
|
|
}
|
|
|
|
|
2017-12-16 12:42:32 +01:00
|
|
|
func getServiceBoolLabel(container dockerData, serviceName string, labelSuffix string, defaultValue bool) bool {
|
|
|
|
if rawValue, ok := getServiceLabels(container, serviceName)[labelSuffix]; ok {
|
|
|
|
value, err := strconv.ParseBool(rawValue)
|
|
|
|
if err == nil {
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return label.GetBoolValue(container.Labels, label.Prefix+labelSuffix, defaultValue)
|
|
|
|
}
|
|
|
|
|
2017-12-16 13:24:07 +01:00
|
|
|
func getServiceIntLabel(container dockerData, serviceName string, labelSuffix string, defaultValue int) int {
|
|
|
|
if rawValue, ok := getServiceLabels(container, serviceName)[labelSuffix]; ok {
|
|
|
|
value, err := strconv.Atoi(rawValue)
|
|
|
|
if err == nil {
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return label.GetIntValue(container.Labels, label.Prefix+labelSuffix, defaultValue)
|
|
|
|
}
|
|
|
|
|
2017-12-02 19:26:44 +01:00
|
|
|
func getServiceLabels(container dockerData, serviceName string) label.ServicePropertyValues {
|
|
|
|
return label.ExtractServiceProperties(container.Labels)[serviceName]
|
|
|
|
}
|