2021-01-19 11:49:50 +01:00
|
|
|
// +build kube
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copyright 2020 Docker Compose CLI authors
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2021-01-29 14:22:22 +01:00
|
|
|
package resources
|
2021-01-19 11:49:50 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/compose-spec/compose-go/types"
|
2021-01-29 11:10:07 +01:00
|
|
|
"github.com/docker/compose-cli/api/compose"
|
2021-01-19 11:49:50 +01:00
|
|
|
apps "k8s.io/api/apps/v1"
|
|
|
|
core "k8s.io/api/core/v1"
|
|
|
|
resource "k8s.io/apimachinery/pkg/api/resource"
|
|
|
|
meta "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
|
|
"k8s.io/apimachinery/pkg/util/intstr"
|
|
|
|
)
|
|
|
|
|
2021-01-28 18:32:32 +01:00
|
|
|
const (
|
|
|
|
clusterIPHeadless = "None"
|
|
|
|
)
|
|
|
|
|
2021-01-26 09:43:12 +01:00
|
|
|
//MapToKubernetesObjects maps compose project to Kubernetes objects
|
2021-01-19 11:49:50 +01:00
|
|
|
func MapToKubernetesObjects(project *types.Project) (map[string]runtime.Object, error) {
|
|
|
|
objects := map[string]runtime.Object{}
|
|
|
|
|
|
|
|
for _, service := range project.Services {
|
|
|
|
svcObject := mapToService(project, service)
|
|
|
|
if svcObject != nil {
|
2021-02-04 13:37:29 +01:00
|
|
|
objects[fmt.Sprintf("%s-service.yaml", getProjectServiceName(project, service))] = svcObject
|
2021-01-19 11:49:50 +01:00
|
|
|
} else {
|
|
|
|
log.Println("Missing port mapping from service config.")
|
|
|
|
}
|
|
|
|
|
|
|
|
if service.Deploy != nil && service.Deploy.Mode == "global" {
|
2021-01-29 11:10:07 +01:00
|
|
|
daemonset, err := mapToDaemonset(project, service)
|
2021-01-19 11:49:50 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-02-04 13:37:29 +01:00
|
|
|
objects[fmt.Sprintf("%s-daemonset.yaml", getProjectServiceName(project, service))] = daemonset
|
2021-01-19 11:49:50 +01:00
|
|
|
} else {
|
2021-01-29 11:10:07 +01:00
|
|
|
deployment, err := mapToDeployment(project, service)
|
2021-01-19 11:49:50 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-02-04 13:37:29 +01:00
|
|
|
objects[fmt.Sprintf("%s-deployment.yaml", getProjectServiceName(project, service))] = deployment
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
for _, vol := range service.Volumes {
|
|
|
|
if vol.Type == "volume" {
|
|
|
|
vol.Source = strings.ReplaceAll(vol.Source, "_", "-")
|
2021-01-29 11:10:07 +01:00
|
|
|
objects[fmt.Sprintf("%s-persistentvolumeclaim.yaml", vol.Source)] = mapToPVC(project, service, vol)
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return objects, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func mapToService(project *types.Project, service types.ServiceConfig) *core.Service {
|
|
|
|
ports := []core.ServicePort{}
|
2021-01-28 18:32:32 +01:00
|
|
|
serviceType := core.ServiceTypeClusterIP
|
|
|
|
clusterIP := ""
|
2021-01-19 11:49:50 +01:00
|
|
|
for _, p := range service.Ports {
|
2021-01-28 18:32:32 +01:00
|
|
|
if p.Published != 0 {
|
|
|
|
serviceType = core.ServiceTypeLoadBalancer
|
|
|
|
}
|
2021-01-19 11:49:50 +01:00
|
|
|
ports = append(ports,
|
|
|
|
core.ServicePort{
|
2021-01-26 09:43:12 +01:00
|
|
|
Name: fmt.Sprintf("%d-%s", p.Target, strings.ToLower(p.Protocol)),
|
2021-01-19 11:49:50 +01:00
|
|
|
Port: int32(p.Target),
|
|
|
|
TargetPort: intstr.FromInt(int(p.Target)),
|
|
|
|
Protocol: toProtocol(p.Protocol),
|
|
|
|
})
|
|
|
|
}
|
2021-01-28 18:32:32 +01:00
|
|
|
if len(ports) == 0 { // headless service
|
|
|
|
clusterIP = clusterIPHeadless
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
return &core.Service{
|
|
|
|
TypeMeta: meta.TypeMeta{
|
|
|
|
Kind: "Service",
|
|
|
|
APIVersion: "v1",
|
|
|
|
},
|
|
|
|
ObjectMeta: meta.ObjectMeta{
|
2021-02-04 13:37:29 +01:00
|
|
|
Name: getProjectServiceName(project, service),
|
2021-01-19 11:49:50 +01:00
|
|
|
},
|
|
|
|
Spec: core.ServiceSpec{
|
2021-01-28 18:32:32 +01:00
|
|
|
ClusterIP: clusterIP,
|
2021-01-29 11:10:07 +01:00
|
|
|
Selector: selectorLabels(project.Name, service.Name),
|
2021-01-28 18:32:32 +01:00
|
|
|
Ports: ports,
|
|
|
|
Type: serviceType,
|
2021-01-19 11:49:50 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-29 11:10:07 +01:00
|
|
|
func mapToDeployment(project *types.Project, service types.ServiceConfig) (*apps.Deployment, error) {
|
|
|
|
labels := selectorLabels(project.Name, service.Name)
|
2021-01-19 11:49:50 +01:00
|
|
|
selector := new(meta.LabelSelector)
|
|
|
|
selector.MatchLabels = make(map[string]string)
|
|
|
|
for key, val := range labels {
|
|
|
|
selector.MatchLabels[key] = val
|
|
|
|
}
|
2021-01-29 11:10:07 +01:00
|
|
|
podTemplate, err := toPodTemplate(project, service, labels)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-01-19 11:49:50 +01:00
|
|
|
return &apps.Deployment{
|
|
|
|
TypeMeta: meta.TypeMeta{
|
|
|
|
Kind: "Deployment",
|
|
|
|
APIVersion: "apps/v1",
|
|
|
|
},
|
|
|
|
ObjectMeta: meta.ObjectMeta{
|
2021-02-04 13:37:29 +01:00
|
|
|
Name: getProjectServiceName(project, service),
|
2021-01-19 11:49:50 +01:00
|
|
|
Labels: labels,
|
|
|
|
},
|
|
|
|
Spec: apps.DeploymentSpec{
|
|
|
|
Selector: selector,
|
|
|
|
Replicas: toReplicas(service.Deploy),
|
|
|
|
Strategy: toDeploymentStrategy(service.Deploy),
|
|
|
|
Template: podTemplate,
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-02-04 13:37:29 +01:00
|
|
|
func getProjectServiceName(project *types.Project, service types.ServiceConfig) string {
|
|
|
|
return fmt.Sprintf("%s-%s", project.Name, service.Name)
|
|
|
|
}
|
|
|
|
|
2021-01-29 11:10:07 +01:00
|
|
|
func selectorLabels(projectName string, serviceName string) map[string]string {
|
|
|
|
return map[string]string{
|
|
|
|
compose.ProjectTag: projectName,
|
|
|
|
compose.ServiceTag: serviceName,
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
2021-01-29 11:10:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func mapToDaemonset(project *types.Project, service types.ServiceConfig) (*apps.DaemonSet, error) {
|
|
|
|
labels := selectorLabels(project.Name, service.Name)
|
2021-01-19 11:49:50 +01:00
|
|
|
podTemplate, err := toPodTemplate(project, service, labels)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &apps.DaemonSet{
|
|
|
|
ObjectMeta: meta.ObjectMeta{
|
2021-02-04 13:37:29 +01:00
|
|
|
Name: getProjectServiceName(project, service),
|
2021-01-19 11:49:50 +01:00
|
|
|
Labels: labels,
|
|
|
|
},
|
|
|
|
Spec: apps.DaemonSetSpec{
|
|
|
|
Template: podTemplate,
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func toReplicas(deploy *types.DeployConfig) *int32 {
|
|
|
|
v := int32(1)
|
|
|
|
if deploy != nil {
|
|
|
|
v = int32(*deploy.Replicas)
|
|
|
|
}
|
|
|
|
return &v
|
|
|
|
}
|
|
|
|
|
|
|
|
func toDeploymentStrategy(deploy *types.DeployConfig) apps.DeploymentStrategy {
|
|
|
|
if deploy == nil || deploy.UpdateConfig == nil {
|
|
|
|
return apps.DeploymentStrategy{
|
|
|
|
Type: apps.RecreateDeploymentStrategyType,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return apps.DeploymentStrategy{
|
|
|
|
Type: apps.RollingUpdateDeploymentStrategyType,
|
|
|
|
RollingUpdate: &apps.RollingUpdateDeployment{
|
|
|
|
MaxUnavailable: &intstr.IntOrString{
|
|
|
|
Type: intstr.Int,
|
|
|
|
IntVal: int32(*deploy.UpdateConfig.Parallelism),
|
|
|
|
},
|
|
|
|
MaxSurge: nil,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-29 11:10:07 +01:00
|
|
|
func mapToPVC(project *types.Project, service types.ServiceConfig, vol types.ServiceVolumeConfig) runtime.Object {
|
2021-01-19 11:49:50 +01:00
|
|
|
rwaccess := core.ReadWriteOnce
|
|
|
|
if vol.ReadOnly {
|
|
|
|
rwaccess = core.ReadOnlyMany
|
|
|
|
}
|
|
|
|
return &core.PersistentVolumeClaim{
|
|
|
|
TypeMeta: meta.TypeMeta{
|
|
|
|
Kind: "PersistentVolumeClaim",
|
|
|
|
APIVersion: "v1",
|
|
|
|
},
|
|
|
|
ObjectMeta: meta.ObjectMeta{
|
|
|
|
Name: vol.Source,
|
2021-01-29 11:10:07 +01:00
|
|
|
Labels: selectorLabels(project.Name, service.Name),
|
2021-01-19 11:49:50 +01:00
|
|
|
},
|
|
|
|
Spec: core.PersistentVolumeClaimSpec{
|
|
|
|
VolumeName: vol.Source,
|
|
|
|
AccessModes: []core.PersistentVolumeAccessMode{rwaccess},
|
|
|
|
Resources: core.ResourceRequirements{
|
|
|
|
Requests: core.ResourceList{
|
|
|
|
core.ResourceStorage: resource.MustParse("100Mi"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// toSecondsOrDefault converts a duration string in seconds and defaults to a
|
|
|
|
// given value if the duration is nil.
|
|
|
|
// The supported units are us, ms, s, m and h.
|
|
|
|
func toSecondsOrDefault(duration *types.Duration, defaultValue int32) int32 { //nolint: unparam
|
|
|
|
if duration == nil {
|
|
|
|
return defaultValue
|
|
|
|
}
|
|
|
|
return int32(time.Duration(*duration).Seconds())
|
|
|
|
}
|