mirror of
				https://github.com/1Panel-dev/1Panel.git
				synced 2025-10-25 06:56:32 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			355 lines
		
	
	
	
		
			9.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			355 lines
		
	
	
	
		
			9.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package docker
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"os"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/1Panel-dev/1Panel/agent/app/model"
 | |
| 	"github.com/1Panel-dev/1Panel/agent/app/task"
 | |
| 	"github.com/1Panel-dev/1Panel/agent/global"
 | |
| 	"github.com/docker/docker/api/types"
 | |
| 	"github.com/docker/docker/api/types/container"
 | |
| 	"github.com/docker/docker/api/types/filters"
 | |
| 	"github.com/docker/docker/api/types/image"
 | |
| 	"github.com/docker/docker/api/types/network"
 | |
| 	"github.com/docker/docker/client"
 | |
| )
 | |
| 
 | |
| func NewDockerClient() (*client.Client, error) {
 | |
| 	var settingItem model.Setting
 | |
| 	_ = global.DB.Where("key = ?", "DockerSockPath").First(&settingItem).Error
 | |
| 	if len(settingItem.Value) == 0 {
 | |
| 		settingItem.Value = "unix:///var/run/docker.sock"
 | |
| 	}
 | |
| 	cli, err := client.NewClientWithOpts(client.FromEnv, client.WithHost(settingItem.Value), client.WithAPIVersionNegotiation())
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return cli, nil
 | |
| }
 | |
| 
 | |
| func NewClient() (Client, error) {
 | |
| 	var settingItem model.Setting
 | |
| 	_ = global.DB.Where("key = ?", "DockerSockPath").First(&settingItem).Error
 | |
| 	if len(settingItem.Value) == 0 {
 | |
| 		settingItem.Value = "unix:///var/run/docker.sock"
 | |
| 	}
 | |
| 	cli, err := client.NewClientWithOpts(client.FromEnv, client.WithHost(settingItem.Value), client.WithAPIVersionNegotiation())
 | |
| 	if err != nil {
 | |
| 		return Client{}, err
 | |
| 	}
 | |
| 
 | |
| 	return Client{
 | |
| 		cli: cli,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func NewClientWithExist(cli *client.Client) Client {
 | |
| 	return Client{
 | |
| 		cli: cli,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type Client struct {
 | |
| 	cli *client.Client
 | |
| }
 | |
| 
 | |
| func (c Client) Close() {
 | |
| 	_ = c.cli.Close()
 | |
| }
 | |
| 
 | |
| func (c Client) ListContainersByName(names []string) ([]types.Container, error) {
 | |
| 	var (
 | |
| 		options  container.ListOptions
 | |
| 		namesMap = make(map[string]bool)
 | |
| 		res      []types.Container
 | |
| 	)
 | |
| 	options.All = true
 | |
| 	if len(names) > 0 {
 | |
| 		var array []filters.KeyValuePair
 | |
| 		for _, n := range names {
 | |
| 			namesMap["/"+n] = true
 | |
| 			array = append(array, filters.Arg("name", n))
 | |
| 		}
 | |
| 		options.Filters = filters.NewArgs(array...)
 | |
| 	}
 | |
| 	containers, err := c.cli.ContainerList(context.Background(), options)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	for _, con := range containers {
 | |
| 		if _, ok := namesMap[con.Names[0]]; ok {
 | |
| 			res = append(res, con)
 | |
| 		}
 | |
| 	}
 | |
| 	return res, nil
 | |
| }
 | |
| func (c Client) ListAllContainers() ([]types.Container, error) {
 | |
| 	var (
 | |
| 		options container.ListOptions
 | |
| 	)
 | |
| 	options.All = true
 | |
| 	containers, err := c.cli.ContainerList(context.Background(), options)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return containers, nil
 | |
| }
 | |
| 
 | |
| func (c Client) CreateNetwork(name string) error {
 | |
| 	_, err := c.cli.NetworkCreate(context.Background(), name, network.CreateOptions{
 | |
| 		Driver:     "bridge",
 | |
| 		EnableIPv6: new(bool),
 | |
| 	})
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (c Client) DeleteImage(imageID string) error {
 | |
| 	if _, err := c.cli.ImageRemove(context.Background(), imageID, image.RemoveOptions{Force: true}); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c Client) GetImageIDByName(imageName string) (string, error) {
 | |
| 	filter := filters.NewArgs()
 | |
| 	filter.Add("reference", imageName)
 | |
| 	list, err := c.cli.ImageList(context.Background(), image.ListOptions{
 | |
| 		Filters: filter,
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	if len(list) > 0 {
 | |
| 		return list[0].ID, nil
 | |
| 	}
 | |
| 	return "", nil
 | |
| }
 | |
| 
 | |
| func (c Client) NetworkExist(name string) bool {
 | |
| 	var options network.ListOptions
 | |
| 	options.Filters = filters.NewArgs(filters.Arg("name", name))
 | |
| 	networks, err := c.cli.NetworkList(context.Background(), options)
 | |
| 	if err != nil {
 | |
| 		return false
 | |
| 	}
 | |
| 	return len(networks) > 0
 | |
| }
 | |
| 
 | |
| func CreateDefaultDockerNetwork() error {
 | |
| 	cli, err := NewClient()
 | |
| 	if err != nil {
 | |
| 		global.LOG.Errorf("init docker client error %s", err.Error())
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	defer cli.Close()
 | |
| 	if !cli.NetworkExist("1panel-network") {
 | |
| 		if err := cli.CreateNetwork("1panel-network"); err != nil {
 | |
| 			global.LOG.Errorf("create default docker network  error %s", err.Error())
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func setLog(id, newLastLine string, task *task.Task) error {
 | |
| 	data, err := os.ReadFile(task.Task.LogFile)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("failed to read file: %v", err)
 | |
| 	}
 | |
| 	lines := strings.Split(string(data), "\n")
 | |
| 	exist := false
 | |
| 	for index, line := range lines {
 | |
| 		if strings.Contains(line, id) {
 | |
| 			lines[index] = newLastLine
 | |
| 			exist = true
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 	if !exist {
 | |
| 		task.Log(newLastLine)
 | |
| 		return nil
 | |
| 	}
 | |
| 	output := strings.Join(lines, "\n")
 | |
| 	_ = os.WriteFile(task.Task.LogFile, []byte(output), os.ModePerm)
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c Client) PullImageWithProcessAndOptions(task *task.Task, imageName string, options image.PullOptions) error {
 | |
| 	out, err := c.cli.ImagePull(context.Background(), imageName, options)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	defer out.Close()
 | |
| 	decoder := json.NewDecoder(out)
 | |
| 	for {
 | |
| 		var progress map[string]interface{}
 | |
| 		if err = decoder.Decode(&progress); err != nil {
 | |
| 			if err == io.EOF {
 | |
| 				break
 | |
| 			}
 | |
| 			return err
 | |
| 		}
 | |
| 		status, _ := progress["status"].(string)
 | |
| 		if status == "Downloading" || status == "Extracting" {
 | |
| 			logProcess(progress, task)
 | |
| 		}
 | |
| 		if status == "Pull complete" || status == "Download complete" {
 | |
| 			id, _ := progress["id"].(string)
 | |
| 			timeStr := time.Now().Format("2006/01/02 15:04:05")
 | |
| 			progressStr := fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, 100.0)
 | |
| 			_ = setLog(id, progressStr, task)
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c Client) PushImageWithProcessAndOptions(task *task.Task, imageName string, options image.PushOptions) error {
 | |
| 	out, err := c.cli.ImagePush(context.Background(), imageName, options)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	defer out.Close()
 | |
| 	decoder := json.NewDecoder(out)
 | |
| 	for {
 | |
| 		var progress map[string]interface{}
 | |
| 		if err = decoder.Decode(&progress); err != nil {
 | |
| 			if err == io.EOF {
 | |
| 				break
 | |
| 			}
 | |
| 			return err
 | |
| 		}
 | |
| 		if msg, ok := progress["errorDetail"]; ok {
 | |
| 			return fmt.Errorf("image push failed, err: %v", msg)
 | |
| 		}
 | |
| 		if msg, ok := progress["error"]; ok {
 | |
| 			return fmt.Errorf("image push failed, err: %v", msg)
 | |
| 		}
 | |
| 		timeStr := time.Now().Format("2006/01/02 15:04:05")
 | |
| 		status, _ := progress["status"].(string)
 | |
| 		switch status {
 | |
| 		case "Pushing":
 | |
| 			id, _ := progress["id"].(string)
 | |
| 			progressDetail, _ := progress["progressDetail"].(map[string]interface{})
 | |
| 			current, _ := progressDetail["current"].(float64)
 | |
| 			progressStr := ""
 | |
| 			total, ok := progressDetail["total"].(float64)
 | |
| 			if ok {
 | |
| 				progressStr = fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, (current/total)*100)
 | |
| 			} else {
 | |
| 				progressStr = fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, current)
 | |
| 			}
 | |
| 
 | |
| 			_ = setLog(id, progressStr, task)
 | |
| 		case "Pushed":
 | |
| 			id, _ := progress["id"].(string)
 | |
| 			progressStr := fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, 100.0)
 | |
| 			_ = setLog(id, progressStr, task)
 | |
| 		default:
 | |
| 			progressStr, _ := json.Marshal(progress)
 | |
| 			task.Log(string(progressStr))
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c Client) BuildImageWithProcessAndOptions(task *task.Task, tar io.ReadCloser, options types.ImageBuildOptions) error {
 | |
| 	out, err := c.cli.ImageBuild(context.Background(), tar, options)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	defer out.Body.Close()
 | |
| 	decoder := json.NewDecoder(out.Body)
 | |
| 	for {
 | |
| 		var progress map[string]interface{}
 | |
| 		if err = decoder.Decode(&progress); err != nil {
 | |
| 			if err == io.EOF {
 | |
| 				break
 | |
| 			}
 | |
| 			return err
 | |
| 		}
 | |
| 		if msg, ok := progress["errorDetail"]; ok {
 | |
| 			return fmt.Errorf("image build failed, err: %v", msg)
 | |
| 		}
 | |
| 		if msg, ok := progress["error"]; ok {
 | |
| 			return fmt.Errorf("image build failed, err: %v", msg)
 | |
| 		}
 | |
| 		timeStr := time.Now().Format("2006/01/02 15:04:05")
 | |
| 		status, _ := progress["status"].(string)
 | |
| 		stream, _ := progress["stream"].(string)
 | |
| 		if len(status) == 0 && len(stream) != 0 {
 | |
| 			if stream != "\n" {
 | |
| 				task.Log(stream)
 | |
| 			}
 | |
| 			continue
 | |
| 		}
 | |
| 		switch status {
 | |
| 		case "Downloading", "Extracting":
 | |
| 			id, _ := progress["id"].(string)
 | |
| 			progressDetail, _ := progress["progressDetail"].(map[string]interface{})
 | |
| 			current, _ := progressDetail["current"].(float64)
 | |
| 			progressStr := ""
 | |
| 			total, ok := progressDetail["total"].(float64)
 | |
| 			if ok {
 | |
| 				progressStr = fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, (current/total)*100)
 | |
| 			} else {
 | |
| 				progressStr = fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, current)
 | |
| 			}
 | |
| 			_ = setLog(id, progressStr, task)
 | |
| 		case "Pull complete", "Download complete", "Verifying Checksum":
 | |
| 			id, _ := progress["id"].(string)
 | |
| 			progressStr := fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, 100.0)
 | |
| 			_ = setLog(id, progressStr, task)
 | |
| 		default:
 | |
| 			progressStr, _ := json.Marshal(progress)
 | |
| 			task.Log(string(progressStr))
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c Client) PullImageWithProcess(task *task.Task, imageName string) error {
 | |
| 	return c.PullImageWithProcessAndOptions(task, imageName, image.PullOptions{})
 | |
| }
 | |
| 
 | |
| func formatBytes(bytes uint64) string {
 | |
| 	const (
 | |
| 		KB = 1024
 | |
| 		MB = 1024 * KB
 | |
| 		GB = 1024 * MB
 | |
| 		TB = 1024 * GB
 | |
| 	)
 | |
| 
 | |
| 	switch {
 | |
| 	case bytes < MB:
 | |
| 		return fmt.Sprintf("%.0fKB", float64(bytes)/KB)
 | |
| 	case bytes < GB:
 | |
| 		return fmt.Sprintf("%.1fMB", float64(bytes)/MB)
 | |
| 	case bytes < TB:
 | |
| 		return fmt.Sprintf("%.1fGB", float64(bytes)/GB)
 | |
| 	default:
 | |
| 		return fmt.Sprintf("%.2fTB", float64(bytes)/TB)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func logProcess(progress map[string]interface{}, task *task.Task) {
 | |
| 	status, _ := progress["status"].(string)
 | |
| 	id, _ := progress["id"].(string)
 | |
| 	progressDetail, _ := progress["progressDetail"].(map[string]interface{})
 | |
| 	current, _ := progressDetail["current"].(float64)
 | |
| 	progressStr := ""
 | |
| 	total, ok := progressDetail["total"].(float64)
 | |
| 	timeStr := time.Now().Format("2006/01/02 15:04:05")
 | |
| 	if ok {
 | |
| 		progressStr = fmt.Sprintf("%s %s [%s] --- %.2f%%", timeStr, status, id, (current/total)*100)
 | |
| 	} else {
 | |
| 		progressStr = fmt.Sprintf("%s %s [%s] --- %s ", timeStr, status, id, formatBytes(uint64(current)))
 | |
| 	}
 | |
| 	_ = setLog(id, progressStr, task)
 | |
| }
 |