mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2025-01-13 11:05:28 +00:00
b15ca52a63
at the moment we compile a script that we can pipe in as single command this is because of the constrains the docker backend gives us. so we move it into the docker backend and eventually get rid of it altogether
194 lines
4.8 KiB
Go
194 lines
4.8 KiB
Go
// Copyright 2022 Woodpecker 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.
|
|
|
|
package docker
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"encoding/json"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/docker/docker/api/types/container"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/backend/common"
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/backend/types"
|
|
)
|
|
|
|
// returns a container configuration.
|
|
func toConfig(step *types.Step) *container.Config {
|
|
config := &container.Config{
|
|
Image: step.Image,
|
|
Labels: step.Labels,
|
|
WorkingDir: step.WorkingDir,
|
|
AttachStdout: true,
|
|
AttachStderr: true,
|
|
}
|
|
|
|
if len(step.Commands) != 0 {
|
|
env, entry, cmd := common.GenerateContainerConf(step.Commands)
|
|
for k, v := range env {
|
|
step.Environment[k] = v
|
|
}
|
|
config.Entrypoint = entry
|
|
config.Cmd = cmd
|
|
}
|
|
|
|
if len(step.Environment) != 0 {
|
|
config.Env = toEnv(step.Environment)
|
|
}
|
|
if len(step.Volumes) != 0 {
|
|
config.Volumes = toVol(step.Volumes)
|
|
}
|
|
return config
|
|
}
|
|
|
|
// returns a container host configuration.
|
|
func toHostConfig(step *types.Step) *container.HostConfig {
|
|
config := &container.HostConfig{
|
|
Resources: container.Resources{
|
|
CPUQuota: step.CPUQuota,
|
|
CPUShares: step.CPUShares,
|
|
CpusetCpus: step.CPUSet,
|
|
Memory: step.MemLimit,
|
|
MemorySwap: step.MemSwapLimit,
|
|
},
|
|
LogConfig: container.LogConfig{
|
|
Type: "json-file",
|
|
},
|
|
Privileged: step.Privileged,
|
|
ShmSize: step.ShmSize,
|
|
Sysctls: step.Sysctls,
|
|
}
|
|
|
|
// if len(step.VolumesFrom) != 0 {
|
|
// config.VolumesFrom = step.VolumesFrom
|
|
// }
|
|
if len(step.NetworkMode) != 0 {
|
|
config.NetworkMode = container.NetworkMode(step.NetworkMode)
|
|
}
|
|
if len(step.IpcMode) != 0 {
|
|
config.IpcMode = container.IpcMode(step.IpcMode)
|
|
}
|
|
if len(step.DNS) != 0 {
|
|
config.DNS = step.DNS
|
|
}
|
|
if len(step.DNSSearch) != 0 {
|
|
config.DNSSearch = step.DNSSearch
|
|
}
|
|
if len(step.ExtraHosts) != 0 {
|
|
config.ExtraHosts = step.ExtraHosts
|
|
}
|
|
if len(step.Devices) != 0 {
|
|
config.Devices = toDev(step.Devices)
|
|
}
|
|
if len(step.Volumes) != 0 {
|
|
config.Binds = step.Volumes
|
|
}
|
|
config.Tmpfs = map[string]string{}
|
|
for _, path := range step.Tmpfs {
|
|
if !strings.Contains(path, ":") {
|
|
config.Tmpfs[path] = ""
|
|
continue
|
|
}
|
|
parts, err := splitVolumeParts(path)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
config.Tmpfs[parts[0]] = parts[1]
|
|
}
|
|
|
|
return config
|
|
}
|
|
|
|
// helper function that converts a slice of volume paths to a set of
|
|
// unique volume names.
|
|
func toVol(paths []string) map[string]struct{} {
|
|
set := map[string]struct{}{}
|
|
for _, path := range paths {
|
|
parts, err := splitVolumeParts(path)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
if len(parts) < 2 {
|
|
continue
|
|
}
|
|
set[parts[1]] = struct{}{}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// helper function that converts a key value map of environment variables to a
|
|
// string slice in key=value format.
|
|
func toEnv(env map[string]string) []string {
|
|
var envs []string
|
|
for k, v := range env {
|
|
envs = append(envs, k+"="+v)
|
|
}
|
|
return envs
|
|
}
|
|
|
|
// helper function that converts a slice of device paths to a slice of
|
|
// container.DeviceMapping.
|
|
func toDev(paths []string) []container.DeviceMapping {
|
|
var devices []container.DeviceMapping
|
|
for _, path := range paths {
|
|
parts, err := splitVolumeParts(path)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
if len(parts) < 2 {
|
|
continue
|
|
}
|
|
if strings.HasSuffix(parts[1], ":ro") || strings.HasSuffix(parts[1], ":rw") {
|
|
parts[1] = parts[1][:len(parts[1])-1]
|
|
}
|
|
devices = append(devices, container.DeviceMapping{
|
|
PathOnHost: parts[0],
|
|
PathInContainer: parts[1],
|
|
CgroupPermissions: "rwm",
|
|
})
|
|
}
|
|
return devices
|
|
}
|
|
|
|
// helper function that serializes the auth configuration as JSON
|
|
// base64 payload.
|
|
func encodeAuthToBase64(authConfig types.Auth) (string, error) {
|
|
buf, err := json.Marshal(authConfig)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return base64.URLEncoding.EncodeToString(buf), nil
|
|
}
|
|
|
|
// helper function that split volume path
|
|
func splitVolumeParts(volumeParts string) ([]string, error) {
|
|
pattern := `^((?:[\w]\:)?[^\:]*)\:((?:[\w]\:)?[^\:]*)(?:\:([rwom]*))?`
|
|
r, err := regexp.Compile(pattern)
|
|
if err != nil {
|
|
return []string{}, err
|
|
}
|
|
if r.MatchString(volumeParts) {
|
|
results := r.FindStringSubmatch(volumeParts)[1:]
|
|
var cleanResults []string
|
|
for _, item := range results {
|
|
if item != "" {
|
|
cleanResults = append(cleanResults, item)
|
|
}
|
|
}
|
|
return cleanResults, nil
|
|
}
|
|
return strings.Split(volumeParts, ":"), nil
|
|
}
|