2022-11-14 11:19:42 +00:00
|
|
|
package service
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
2023-01-05 09:29:27 +00:00
|
|
|
"context"
|
2022-11-14 11:19:42 +00:00
|
|
|
"encoding/json"
|
2024-10-14 02:36:55 +00:00
|
|
|
"fmt"
|
2022-11-14 11:19:42 +00:00
|
|
|
"os"
|
2023-01-05 09:29:27 +00:00
|
|
|
"path"
|
2022-11-14 11:19:42 +00:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/1Panel-dev/1Panel/backend/app/dto"
|
2022-11-18 08:14:23 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/constant"
|
2024-09-23 06:31:14 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/global"
|
2023-02-14 06:19:26 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/utils/cmd"
|
2023-01-05 09:29:27 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/utils/docker"
|
2024-09-23 06:31:14 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/utils/systemctl"
|
2022-11-18 08:14:23 +00:00
|
|
|
"github.com/pkg/errors"
|
2022-11-14 11:19:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type DockerService struct{}
|
|
|
|
|
|
|
|
type IDockerService interface {
|
2023-05-29 03:24:28 +00:00
|
|
|
UpdateConf(req dto.SettingUpdate) error
|
|
|
|
UpdateLogOption(req dto.LogOption) error
|
2023-11-21 14:14:07 +00:00
|
|
|
UpdateIpv6Option(req dto.Ipv6Option) error
|
2022-11-14 11:19:42 +00:00
|
|
|
UpdateConfByFile(info dto.DaemonJsonUpdateByFile) error
|
2022-12-07 09:28:14 +00:00
|
|
|
LoadDockerStatus() string
|
|
|
|
LoadDockerConf() *dto.DaemonJsonConf
|
|
|
|
OperateDocker(req dto.DockerOperation) error
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewIDockerService() IDockerService {
|
|
|
|
return &DockerService{}
|
|
|
|
}
|
|
|
|
|
|
|
|
type daemonJsonItem struct {
|
2023-11-21 14:14:07 +00:00
|
|
|
Status string `json:"status"`
|
|
|
|
Mirrors []string `json:"registry-mirrors"`
|
|
|
|
Registries []string `json:"insecure-registries"`
|
|
|
|
LiveRestore bool `json:"live-restore"`
|
|
|
|
Ipv6 bool `json:"ipv6"`
|
|
|
|
FixedCidrV6 string `json:"fixed-cidr-v6"`
|
|
|
|
Ip6Tables bool `json:"ip6tables"`
|
|
|
|
Experimental bool `json:"experimental"`
|
|
|
|
IPTables bool `json:"iptables"`
|
|
|
|
ExecOpts []string `json:"exec-opts"`
|
|
|
|
LogOption logOption `json:"log-opts"`
|
2023-05-18 10:38:19 +00:00
|
|
|
}
|
|
|
|
type logOption struct {
|
|
|
|
LogMaxSize string `json:"max-size"`
|
|
|
|
LogMaxFile string `json:"max-file"`
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
|
|
|
|
2022-12-07 09:28:14 +00:00
|
|
|
func (u *DockerService) LoadDockerStatus() string {
|
2023-05-18 08:46:13 +00:00
|
|
|
client, err := docker.NewDockerClient()
|
|
|
|
if err != nil {
|
|
|
|
return constant.Stopped
|
|
|
|
}
|
2024-04-25 08:45:12 +00:00
|
|
|
defer client.Close()
|
2023-05-18 08:46:13 +00:00
|
|
|
if _, err := client.Ping(context.Background()); err != nil {
|
|
|
|
return constant.Stopped
|
2023-04-27 02:24:14 +00:00
|
|
|
}
|
2022-12-08 03:56:07 +00:00
|
|
|
|
2023-05-18 08:46:13 +00:00
|
|
|
return constant.StatusRunning
|
2022-12-07 09:28:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *DockerService) LoadDockerConf() *dto.DaemonJsonConf {
|
2023-05-18 08:46:13 +00:00
|
|
|
ctx := context.Background()
|
2023-04-10 09:04:23 +00:00
|
|
|
var data dto.DaemonJsonConf
|
|
|
|
data.IPTables = true
|
|
|
|
data.Status = constant.StatusRunning
|
|
|
|
data.Version = "-"
|
2023-01-05 09:29:27 +00:00
|
|
|
client, err := docker.NewDockerClient()
|
2023-05-18 08:46:13 +00:00
|
|
|
if err != nil {
|
|
|
|
data.Status = constant.Stopped
|
|
|
|
} else {
|
2024-04-25 08:45:12 +00:00
|
|
|
defer client.Close()
|
2023-05-18 08:46:13 +00:00
|
|
|
if _, err := client.Ping(ctx); err != nil {
|
|
|
|
data.Status = constant.Stopped
|
|
|
|
}
|
2023-01-05 09:29:27 +00:00
|
|
|
itemVersion, err := client.ServerVersion(ctx)
|
|
|
|
if err == nil {
|
2023-04-10 09:04:23 +00:00
|
|
|
data.Version = itemVersion.Version
|
2023-01-05 09:29:27 +00:00
|
|
|
}
|
2022-12-07 09:28:14 +00:00
|
|
|
}
|
2023-05-18 08:46:13 +00:00
|
|
|
data.IsSwarm = false
|
|
|
|
stdout2, _ := cmd.Exec("docker info | grep Swarm")
|
|
|
|
if string(stdout2) == " Swarm: active\n" {
|
|
|
|
data.IsSwarm = true
|
|
|
|
}
|
2023-01-05 09:29:27 +00:00
|
|
|
if _, err := os.Stat(constant.DaemonJsonPath); err != nil {
|
2023-04-10 09:04:23 +00:00
|
|
|
return &data
|
2022-12-07 09:28:14 +00:00
|
|
|
}
|
2023-04-07 03:30:10 +00:00
|
|
|
file, err := os.ReadFile(constant.DaemonJsonPath)
|
2022-12-07 09:28:14 +00:00
|
|
|
if err != nil {
|
2023-04-10 09:04:23 +00:00
|
|
|
return &data
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
|
|
|
var conf daemonJsonItem
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap := make(map[string]interface{})
|
|
|
|
if err := json.Unmarshal(file, &daemonMap); err != nil {
|
2023-04-10 09:04:23 +00:00
|
|
|
return &data
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
2023-05-30 07:30:57 +00:00
|
|
|
arr, err := json.Marshal(daemonMap)
|
2022-11-14 11:19:42 +00:00
|
|
|
if err != nil {
|
2023-04-10 09:04:23 +00:00
|
|
|
return &data
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
|
|
|
if err := json.Unmarshal(arr, &conf); err != nil {
|
2023-04-10 09:04:23 +00:00
|
|
|
return &data
|
2023-04-07 09:44:15 +00:00
|
|
|
}
|
2023-05-30 07:30:57 +00:00
|
|
|
if _, ok := daemonMap["iptables"]; !ok {
|
2023-04-07 09:44:15 +00:00
|
|
|
conf.IPTables = true
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
2023-04-10 09:04:23 +00:00
|
|
|
data.CgroupDriver = "cgroupfs"
|
2022-11-14 11:19:42 +00:00
|
|
|
for _, opt := range conf.ExecOpts {
|
|
|
|
if strings.HasPrefix(opt, "native.cgroupdriver=") {
|
2023-04-10 09:04:23 +00:00
|
|
|
data.CgroupDriver = strings.ReplaceAll(opt, "native.cgroupdriver=", "")
|
2022-11-14 11:19:42 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2023-11-21 14:14:07 +00:00
|
|
|
data.Ipv6 = conf.Ipv6
|
|
|
|
data.FixedCidrV6 = conf.FixedCidrV6
|
|
|
|
data.Ip6Tables = conf.Ip6Tables
|
|
|
|
data.Experimental = conf.Experimental
|
2023-05-18 10:38:19 +00:00
|
|
|
data.LogMaxSize = conf.LogOption.LogMaxSize
|
|
|
|
data.LogMaxFile = conf.LogOption.LogMaxFile
|
2023-04-10 09:04:23 +00:00
|
|
|
data.Mirrors = conf.Mirrors
|
|
|
|
data.Registries = conf.Registries
|
|
|
|
data.IPTables = conf.IPTables
|
|
|
|
data.LiveRestore = conf.LiveRestore
|
2022-12-07 09:28:14 +00:00
|
|
|
return &data
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
|
|
|
|
2023-05-29 03:24:28 +00:00
|
|
|
func (u *DockerService) UpdateConf(req dto.SettingUpdate) error {
|
2024-06-24 07:29:19 +00:00
|
|
|
err := createIfNotExistDaemonJsonFile()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2022-12-07 09:28:14 +00:00
|
|
|
}
|
2023-04-07 03:30:10 +00:00
|
|
|
file, err := os.ReadFile(constant.DaemonJsonPath)
|
2022-12-07 09:28:14 +00:00
|
|
|
if err != nil {
|
2022-11-14 11:19:42 +00:00
|
|
|
return err
|
|
|
|
}
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap := make(map[string]interface{})
|
|
|
|
_ = json.Unmarshal(file, &daemonMap)
|
2022-12-07 09:28:14 +00:00
|
|
|
|
2023-05-29 03:24:28 +00:00
|
|
|
switch req.Key {
|
|
|
|
case "Registries":
|
2023-08-02 08:47:30 +00:00
|
|
|
req.Value = strings.TrimSuffix(req.Value, ",")
|
2023-05-29 03:24:28 +00:00
|
|
|
if len(req.Value) == 0 {
|
2023-05-30 07:30:57 +00:00
|
|
|
delete(daemonMap, "insecure-registries")
|
2023-05-29 03:24:28 +00:00
|
|
|
} else {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["insecure-registries"] = strings.Split(req.Value, ",")
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
|
|
|
case "Mirrors":
|
2023-08-02 08:47:30 +00:00
|
|
|
req.Value = strings.TrimSuffix(req.Value, ",")
|
2023-05-29 03:24:28 +00:00
|
|
|
if len(req.Value) == 0 {
|
2023-05-30 07:30:57 +00:00
|
|
|
delete(daemonMap, "registry-mirrors")
|
2023-05-29 03:24:28 +00:00
|
|
|
} else {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["registry-mirrors"] = strings.Split(req.Value, ",")
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
2023-11-21 14:14:07 +00:00
|
|
|
case "Ipv6":
|
|
|
|
if req.Value == "disable" {
|
|
|
|
delete(daemonMap, "ipv6")
|
|
|
|
delete(daemonMap, "fixed-cidr-v6")
|
|
|
|
delete(daemonMap, "ip6tables")
|
|
|
|
delete(daemonMap, "experimental")
|
|
|
|
}
|
2023-05-29 03:24:28 +00:00
|
|
|
case "LogOption":
|
|
|
|
if req.Value == "disable" {
|
2023-05-30 07:30:57 +00:00
|
|
|
delete(daemonMap, "log-opts")
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
|
|
|
case "LiveRestore":
|
|
|
|
if req.Value == "disable" {
|
2023-05-30 07:30:57 +00:00
|
|
|
delete(daemonMap, "live-restore")
|
2023-05-29 03:24:28 +00:00
|
|
|
} else {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["live-restore"] = true
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
|
|
|
case "IPtables":
|
|
|
|
if req.Value == "enable" {
|
2023-05-30 07:30:57 +00:00
|
|
|
delete(daemonMap, "iptables")
|
2023-05-29 03:24:28 +00:00
|
|
|
} else {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["iptables"] = false
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
2024-02-17 14:42:09 +00:00
|
|
|
case "Driver":
|
2023-05-30 07:30:57 +00:00
|
|
|
if opts, ok := daemonMap["exec-opts"]; ok {
|
2023-05-29 03:24:28 +00:00
|
|
|
if optsValue, isArray := opts.([]interface{}); isArray {
|
|
|
|
for i := 0; i < len(optsValue); i++ {
|
|
|
|
if opt, isStr := optsValue[i].(string); isStr {
|
|
|
|
if strings.HasPrefix(opt, "native.cgroupdriver=") {
|
|
|
|
optsValue[i] = "native.cgroupdriver=" + req.Value
|
|
|
|
break
|
|
|
|
}
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-05-29 03:24:28 +00:00
|
|
|
} else {
|
|
|
|
if req.Value == "systemd" {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["exec-opts"] = []string{"native.cgroupdriver=systemd"}
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
2024-11-08 10:19:48 +00:00
|
|
|
case "http-proxy", "https-proxy":
|
|
|
|
delete(daemonMap, "proxies")
|
|
|
|
if len(req.Value) > 0 {
|
|
|
|
proxies := map[string]interface{}{
|
|
|
|
req.Key: req.Value,
|
|
|
|
}
|
|
|
|
daemonMap["proxies"] = proxies
|
|
|
|
}
|
2024-11-13 06:50:00 +00:00
|
|
|
case "socks5-proxy", "close-proxy":
|
2024-11-08 10:19:48 +00:00
|
|
|
delete(daemonMap, "proxies")
|
|
|
|
if len(req.Value) > 0 {
|
|
|
|
proxies := map[string]interface{}{
|
|
|
|
"http-proxy": req.Value,
|
|
|
|
"https-proxy": req.Value,
|
|
|
|
}
|
|
|
|
daemonMap["proxies"] = proxies
|
|
|
|
}
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
2023-05-30 07:30:57 +00:00
|
|
|
if len(daemonMap) == 0 {
|
2023-05-29 03:24:28 +00:00
|
|
|
_ = os.Remove(constant.DaemonJsonPath)
|
2024-10-17 10:03:31 +00:00
|
|
|
if err := restartDocker(); err != nil {
|
|
|
|
return err
|
2024-08-23 10:24:58 +00:00
|
|
|
}
|
2023-05-29 03:24:28 +00:00
|
|
|
return nil
|
|
|
|
}
|
2023-05-30 07:30:57 +00:00
|
|
|
newJson, err := json.MarshalIndent(daemonMap, "", "\t")
|
2023-05-29 03:24:28 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := os.WriteFile(constant.DaemonJsonPath, newJson, 0640); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-10-14 02:36:55 +00:00
|
|
|
if err := validateDockerConfig(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-05-29 03:24:28 +00:00
|
|
|
|
2024-10-17 10:03:31 +00:00
|
|
|
if err := restartDocker(); err != nil {
|
|
|
|
return err
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2024-06-24 07:29:19 +00:00
|
|
|
func createIfNotExistDaemonJsonFile() error {
|
2023-05-29 03:24:28 +00:00
|
|
|
if _, err := os.Stat(constant.DaemonJsonPath); err != nil && os.IsNotExist(err) {
|
|
|
|
if err = os.MkdirAll(path.Dir(constant.DaemonJsonPath), os.ModePerm); err != nil {
|
|
|
|
return err
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
2024-06-24 07:29:19 +00:00
|
|
|
var daemonFile *os.File
|
|
|
|
daemonFile, err = os.Create(constant.DaemonJsonPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer daemonFile.Close()
|
2023-05-29 03:24:28 +00:00
|
|
|
}
|
2024-06-24 07:29:19 +00:00
|
|
|
return nil
|
|
|
|
}
|
2023-05-29 03:24:28 +00:00
|
|
|
|
2024-06-24 07:29:19 +00:00
|
|
|
func (u *DockerService) UpdateLogOption(req dto.LogOption) error {
|
|
|
|
err := createIfNotExistDaemonJsonFile()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-05-29 03:24:28 +00:00
|
|
|
file, err := os.ReadFile(constant.DaemonJsonPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2022-11-14 11:19:42 +00:00
|
|
|
}
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap := make(map[string]interface{})
|
|
|
|
_ = json.Unmarshal(file, &daemonMap)
|
2023-05-29 03:24:28 +00:00
|
|
|
|
2023-05-30 07:30:57 +00:00
|
|
|
changeLogOption(daemonMap, req.LogMaxFile, req.LogMaxSize)
|
|
|
|
if len(daemonMap) == 0 {
|
2023-04-09 15:58:13 +00:00
|
|
|
_ = os.Remove(constant.DaemonJsonPath)
|
|
|
|
return nil
|
|
|
|
}
|
2023-05-30 07:30:57 +00:00
|
|
|
newJson, err := json.MarshalIndent(daemonMap, "", "\t")
|
2022-11-14 11:19:42 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-04-07 03:30:10 +00:00
|
|
|
if err := os.WriteFile(constant.DaemonJsonPath, newJson, 0640); err != nil {
|
2022-11-14 11:19:42 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-11-18 08:14:23 +00:00
|
|
|
|
2024-10-14 02:36:55 +00:00
|
|
|
if err := validateDockerConfig(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-10-17 10:03:31 +00:00
|
|
|
if err := restartDocker(); err != nil {
|
|
|
|
return err
|
2022-11-18 08:14:23 +00:00
|
|
|
}
|
2022-11-14 11:19:42 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:14:07 +00:00
|
|
|
func (u *DockerService) UpdateIpv6Option(req dto.Ipv6Option) error {
|
2024-06-24 07:29:19 +00:00
|
|
|
err := createIfNotExistDaemonJsonFile()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2023-11-21 14:14:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
file, err := os.ReadFile(constant.DaemonJsonPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
daemonMap := make(map[string]interface{})
|
|
|
|
_ = json.Unmarshal(file, &daemonMap)
|
|
|
|
|
|
|
|
daemonMap["ipv6"] = true
|
|
|
|
daemonMap["fixed-cidr-v6"] = req.FixedCidrV6
|
|
|
|
if req.Ip6Tables {
|
|
|
|
daemonMap["ip6tables"] = req.Ip6Tables
|
|
|
|
}
|
|
|
|
if req.Experimental {
|
|
|
|
daemonMap["experimental"] = req.Experimental
|
|
|
|
}
|
|
|
|
if len(daemonMap) == 0 {
|
|
|
|
_ = os.Remove(constant.DaemonJsonPath)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
newJson, err := json.MarshalIndent(daemonMap, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := os.WriteFile(constant.DaemonJsonPath, newJson, 0640); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-10-14 02:36:55 +00:00
|
|
|
if err := validateDockerConfig(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-10-17 10:03:31 +00:00
|
|
|
if err := restartDocker(); err != nil {
|
|
|
|
return err
|
2023-11-21 14:14:07 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-11-14 11:19:42 +00:00
|
|
|
func (u *DockerService) UpdateConfByFile(req dto.DaemonJsonUpdateByFile) error {
|
2023-04-09 15:58:13 +00:00
|
|
|
if len(req.File) == 0 {
|
|
|
|
_ = os.Remove(constant.DaemonJsonPath)
|
2024-10-17 10:03:31 +00:00
|
|
|
if err := restartDocker(); err != nil {
|
|
|
|
return err
|
2024-08-23 10:24:58 +00:00
|
|
|
}
|
2023-04-09 15:58:13 +00:00
|
|
|
return nil
|
|
|
|
}
|
2024-06-24 07:29:19 +00:00
|
|
|
err := createIfNotExistDaemonJsonFile()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2023-04-07 09:44:15 +00:00
|
|
|
}
|
2023-03-10 10:05:02 +00:00
|
|
|
file, err := os.OpenFile(constant.DaemonJsonPath, os.O_WRONLY|os.O_TRUNC, 0640)
|
2022-11-14 11:19:42 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer file.Close()
|
|
|
|
write := bufio.NewWriter(file)
|
|
|
|
_, _ = write.WriteString(req.File)
|
|
|
|
write.Flush()
|
|
|
|
|
2024-10-14 02:36:55 +00:00
|
|
|
if err := validateDockerConfig(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-10-17 10:03:31 +00:00
|
|
|
if err := restartDocker(); err != nil {
|
|
|
|
return err
|
2022-11-18 08:14:23 +00:00
|
|
|
}
|
2022-11-14 11:19:42 +00:00
|
|
|
return nil
|
|
|
|
}
|
2022-12-07 09:28:14 +00:00
|
|
|
|
|
|
|
func (u *DockerService) OperateDocker(req dto.DockerOperation) error {
|
2023-03-11 15:44:16 +00:00
|
|
|
service := "docker"
|
2024-09-23 06:31:14 +00:00
|
|
|
sudo := cmd.SudoHandleCmd()
|
2024-10-17 10:03:31 +00:00
|
|
|
dockerCmd, err := getDockerRestartCommand()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-11 15:44:16 +00:00
|
|
|
if req.Operation == "stop" {
|
2024-09-23 06:31:14 +00:00
|
|
|
isSocketActive, _ := systemctl.IsActive("docker.socket")
|
|
|
|
if isSocketActive {
|
|
|
|
std, err := cmd.Execf("%s systemctl stop docker.socket", sudo)
|
|
|
|
if err != nil {
|
|
|
|
global.LOG.Errorf("handle systemctl stop docker.socket failed, err: %v", std)
|
|
|
|
}
|
|
|
|
}
|
2023-03-11 15:44:16 +00:00
|
|
|
}
|
2024-10-14 02:36:55 +00:00
|
|
|
|
|
|
|
if req.Operation == "restart" {
|
|
|
|
if err := validateDockerConfig(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-17 10:03:31 +00:00
|
|
|
stdout, err := cmd.Execf("%s %s %s", dockerCmd, req.Operation, service)
|
2022-12-07 09:28:14 +00:00
|
|
|
if err != nil {
|
2024-10-17 10:03:31 +00:00
|
|
|
return errors.New(stdout)
|
2022-12-07 09:28:14 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2023-05-18 10:38:19 +00:00
|
|
|
|
2023-05-30 07:30:57 +00:00
|
|
|
func changeLogOption(daemonMap map[string]interface{}, logMaxFile, logMaxSize string) {
|
|
|
|
if opts, ok := daemonMap["log-opts"]; ok {
|
2023-05-18 10:38:19 +00:00
|
|
|
if len(logMaxFile) != 0 || len(logMaxSize) != 0 {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["log-driver"] = "json-file"
|
2023-05-18 10:38:19 +00:00
|
|
|
}
|
|
|
|
optsMap, isMap := opts.(map[string]interface{})
|
|
|
|
if isMap {
|
|
|
|
if len(logMaxFile) != 0 {
|
|
|
|
optsMap["max-file"] = logMaxFile
|
|
|
|
} else {
|
|
|
|
delete(optsMap, "max-file")
|
|
|
|
}
|
|
|
|
if len(logMaxSize) != 0 {
|
|
|
|
optsMap["max-size"] = logMaxSize
|
|
|
|
} else {
|
|
|
|
delete(optsMap, "max-size")
|
|
|
|
}
|
|
|
|
if len(optsMap) == 0 {
|
2023-05-30 07:30:57 +00:00
|
|
|
delete(daemonMap, "log-opts")
|
2023-05-18 10:38:19 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
optsMap := make(map[string]interface{})
|
|
|
|
if len(logMaxFile) != 0 {
|
|
|
|
optsMap["max-file"] = logMaxFile
|
|
|
|
}
|
|
|
|
if len(logMaxSize) != 0 {
|
|
|
|
optsMap["max-size"] = logMaxSize
|
|
|
|
}
|
|
|
|
if len(optsMap) != 0 {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["log-opts"] = optsMap
|
2023-05-18 10:38:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if len(logMaxFile) != 0 || len(logMaxSize) != 0 {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["log-driver"] = "json-file"
|
2023-05-18 10:38:19 +00:00
|
|
|
}
|
|
|
|
optsMap := make(map[string]interface{})
|
|
|
|
if len(logMaxFile) != 0 {
|
|
|
|
optsMap["max-file"] = logMaxFile
|
|
|
|
}
|
|
|
|
if len(logMaxSize) != 0 {
|
|
|
|
optsMap["max-size"] = logMaxSize
|
|
|
|
}
|
|
|
|
if len(optsMap) != 0 {
|
2023-05-30 07:30:57 +00:00
|
|
|
daemonMap["log-opts"] = optsMap
|
2023-05-18 10:38:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-10-14 02:36:55 +00:00
|
|
|
|
|
|
|
func validateDockerConfig() error {
|
2024-10-22 09:42:28 +00:00
|
|
|
if !cmd.Which("dockerd") {
|
|
|
|
return nil
|
|
|
|
}
|
2024-10-14 02:36:55 +00:00
|
|
|
stdout, err := cmd.Exec("dockerd --validate")
|
2024-10-23 10:34:19 +00:00
|
|
|
if strings.Contains(stdout, "unknown flag: --validate") {
|
|
|
|
return nil
|
|
|
|
}
|
2024-10-14 02:36:55 +00:00
|
|
|
if err != nil || (stdout != "" && strings.TrimSpace(stdout) != "configuration OK") {
|
|
|
|
return fmt.Errorf("Docker configuration validation failed, err: %v", stdout)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2024-10-17 10:03:31 +00:00
|
|
|
|
|
|
|
func getDockerRestartCommand() (string, error) {
|
|
|
|
stdout, err := cmd.Exec("which docker")
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("failed to find docker: %v", err)
|
|
|
|
}
|
|
|
|
dockerPath := stdout
|
|
|
|
if strings.Contains(dockerPath, "snap") {
|
|
|
|
return "snap", nil
|
|
|
|
}
|
|
|
|
return "systemctl", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func restartDocker() error {
|
|
|
|
restartCmd, err := getDockerRestartCommand()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
stdout, err := cmd.Execf("%s restart docker", restartCmd)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to restart Docker: %s", stdout)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|