mirror of
https://github.com/linka-cloud/d2vm.git
synced 2024-11-24 16:46:24 +00:00
Adphi
d54b3f9a2c
* build / convert: add `--platform` flag to support linux/amd64 & linux/arm64 * build: add `--pull` flag * run/hetzner: add `--type` flag to select server type * run/hetzner: add `--location` flag to select server location Signed-off-by: Adphi <philippe.adrien.nousse@gmail.com>
171 lines
4.1 KiB
Go
171 lines
4.1 KiB
Go
// Copyright 2022 Linka Cloud All rights reserved.
|
|
//
|
|
// 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 (
|
|
"bufio"
|
|
"context"
|
|
_ "embed"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"go.linka.cloud/d2vm/pkg/exec"
|
|
)
|
|
|
|
func dockerSocket() string {
|
|
if runtime.GOOS == "windows" {
|
|
return "//var/run/docker.sock"
|
|
}
|
|
return "/var/run/docker.sock"
|
|
}
|
|
|
|
func FormatImgName(name string) string {
|
|
s := strings.Replace(name, ":", "-", -1)
|
|
s = strings.Replace(s, "/", "_", -1)
|
|
return s
|
|
}
|
|
|
|
func Cmd(ctx context.Context, args ...string) error {
|
|
return exec.Run(ctx, "docker", args...)
|
|
}
|
|
|
|
func CmdOut(ctx context.Context, args ...string) (string, string, error) {
|
|
return exec.RunOut(ctx, "docker", args...)
|
|
}
|
|
|
|
func Build(ctx context.Context, pull bool, tag, dockerfile, dir, platform string, buildArgs ...string) error {
|
|
if dockerfile == "" {
|
|
dockerfile = filepath.Join(dir, "Dockerfile")
|
|
}
|
|
args := []string{"image", "build", "-t", tag, "-f", dockerfile, "--platform", platform}
|
|
if pull {
|
|
args = append(args, "--pull")
|
|
}
|
|
for _, v := range buildArgs {
|
|
args = append(args, "--build-arg", v)
|
|
}
|
|
args = append(args, dir)
|
|
return Cmd(ctx, args...)
|
|
}
|
|
|
|
func Tag(ctx context.Context, img string, tags ...string) error {
|
|
if len(tags) == 0 {
|
|
return fmt.Errorf("no tags specified")
|
|
}
|
|
args := []string{"image", "tag"}
|
|
for _, tag := range tags {
|
|
if err := Cmd(ctx, append(args, img, tag)...); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func Remove(ctx context.Context, tag string) error {
|
|
return Cmd(ctx, "image", "rm", tag)
|
|
}
|
|
|
|
func ImageList(ctx context.Context, tag string) ([]string, error) {
|
|
o, _, err := CmdOut(ctx, "image", "ls", "--format={{ .Repository }}:{{ .Tag }}", tag)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
s := bufio.NewScanner(strings.NewReader(o))
|
|
var imgs []string
|
|
for s.Scan() {
|
|
imgs = append(imgs, s.Text())
|
|
}
|
|
return imgs, s.Err()
|
|
}
|
|
|
|
func ImageSave(ctx context.Context, tag, file string) error {
|
|
return Cmd(ctx, "image", "save", "-o", file, tag)
|
|
}
|
|
|
|
func Pull(ctx context.Context, platform, tag string) error {
|
|
return Cmd(ctx, "image", "pull", "--platform", platform, tag)
|
|
}
|
|
|
|
func Push(ctx context.Context, tag string) error {
|
|
return Cmd(ctx, "image", "push", tag)
|
|
}
|
|
|
|
func RunInteractiveAndRemove(ctx context.Context, args ...string) error {
|
|
cmd := exec.CommandContext(ctx, "docker", append([]string{"run", "--rm", "-it"}, args...)...)
|
|
cmd.Stdin = os.Stdin
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
return cmd.Run()
|
|
}
|
|
|
|
func RunAndRemove(ctx context.Context, args ...string) error {
|
|
logrus.Tracef("running 'docker run --rm %s'", strings.Join(args, " "))
|
|
return Cmd(ctx, append([]string{"run", "--rm"}, args...)...)
|
|
}
|
|
|
|
func RunD2VM(ctx context.Context, image, version, in, out, cmd string, args ...string) error {
|
|
pwd, err := os.Getwd()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if in == "" {
|
|
in = pwd
|
|
}
|
|
if out == "" {
|
|
out = pwd
|
|
}
|
|
if image == "" {
|
|
image = "linkacloud/d2vm"
|
|
}
|
|
if version == "" {
|
|
version = "latest"
|
|
}
|
|
a := []string{"run", "--rm"}
|
|
|
|
interactive := isInteractive()
|
|
|
|
if interactive {
|
|
a = append(a, "-i", "-t")
|
|
}
|
|
a = append(a,
|
|
"--privileged",
|
|
"-e",
|
|
// yes... it is kind of a dirty hack
|
|
fmt.Sprintf("SUDO_UID=%d", os.Getuid()),
|
|
"-v",
|
|
fmt.Sprintf("%s:/var/run/docker.sock", dockerSocket()),
|
|
"-v",
|
|
fmt.Sprintf("%s:/in", in),
|
|
"-v",
|
|
fmt.Sprintf("%s:/out", out),
|
|
"-w",
|
|
"/d2vm",
|
|
fmt.Sprintf("%s:%s", image, version),
|
|
cmd,
|
|
)
|
|
c := exec.CommandContext(ctx, "docker", append(a, args...)...)
|
|
if interactive {
|
|
c.Stdin = os.Stdin
|
|
}
|
|
c.Stdout = os.Stdout
|
|
c.Stderr = os.Stderr
|
|
return c.Run()
|
|
}
|