2019-11-05 16:37:21 +00:00
|
|
|
// Copyright © 2019 Ettore Di Giacinto <mudler@gentoo.org>
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation; either version 2 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License along
|
|
|
|
// with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
package backend
|
|
|
|
|
|
|
|
import (
|
2019-12-02 15:25:01 +00:00
|
|
|
"os"
|
2019-11-05 16:37:21 +00:00
|
|
|
"os/exec"
|
2021-01-23 21:01:29 +00:00
|
|
|
"strings"
|
2019-11-05 16:37:21 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
"github.com/mudler/luet/pkg/api/core/types"
|
2021-03-16 13:46:28 +00:00
|
|
|
bus "github.com/mudler/luet/pkg/bus"
|
|
|
|
|
2019-11-05 16:37:21 +00:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
type SimpleImg struct {
|
|
|
|
ctx *types.Context
|
|
|
|
}
|
2019-11-05 16:37:21 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
func NewSimpleImgBackend(ctx *types.Context) *SimpleImg {
|
|
|
|
return &SimpleImg{ctx: ctx}
|
2019-11-05 16:37:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Missing still: labels, and build args expansion
|
2021-10-20 22:13:02 +00:00
|
|
|
func (s *SimpleImg) BuildImage(opts Options) error {
|
2019-11-08 17:29:51 +00:00
|
|
|
name := opts.ImageName
|
2021-03-16 13:46:28 +00:00
|
|
|
bus.Manager.Publish(bus.EventImagePreBuild, opts)
|
2021-01-18 10:05:46 +00:00
|
|
|
|
2021-02-22 12:48:26 +00:00
|
|
|
buildarg := genBuildCommand(opts)
|
2021-02-01 18:10:16 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Info(":tea: Building image " + name)
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2019-12-02 15:25:01 +00:00
|
|
|
cmd := exec.Command("img", buildarg...)
|
2021-02-22 12:48:26 +00:00
|
|
|
cmd.Dir = opts.SourcePath
|
2021-10-20 22:13:02 +00:00
|
|
|
err := runCommand(s.ctx, cmd)
|
2019-11-05 16:37:21 +00:00
|
|
|
if err != nil {
|
2021-02-01 18:10:16 +00:00
|
|
|
return err
|
2019-11-05 16:37:21 +00:00
|
|
|
}
|
2021-03-16 13:46:28 +00:00
|
|
|
bus.Manager.Publish(bus.EventImagePostBuild, opts)
|
2021-02-01 18:10:16 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Info(":tea: Building image " + name + " done")
|
2021-02-01 18:10:16 +00:00
|
|
|
|
2019-11-05 16:37:21 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
func (s *SimpleImg) RemoveImage(opts Options) error {
|
2019-11-08 17:29:51 +00:00
|
|
|
name := opts.ImageName
|
|
|
|
buildarg := []string{"rm", name}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Spinner()
|
|
|
|
defer s.ctx.SpinnerStop()
|
2019-11-08 17:29:51 +00:00
|
|
|
out, err := exec.Command("img", buildarg...).CombinedOutput()
|
2019-11-05 16:37:21 +00:00
|
|
|
if err != nil {
|
2020-12-11 22:03:56 +00:00
|
|
|
return errors.Wrap(err, "Failed removing image: "+string(out))
|
2019-11-05 16:37:21 +00:00
|
|
|
}
|
2019-12-02 15:25:01 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Info(":tea: Image " + name + " removed")
|
2019-11-05 16:37:21 +00:00
|
|
|
return nil
|
2019-11-08 17:29:51 +00:00
|
|
|
}
|
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
func (s *SimpleImg) DownloadImage(opts Options) error {
|
2019-11-13 08:42:52 +00:00
|
|
|
name := opts.ImageName
|
2021-03-16 13:46:28 +00:00
|
|
|
bus.Manager.Publish(bus.EventImagePrePull, opts)
|
|
|
|
|
2019-11-13 08:42:52 +00:00
|
|
|
buildarg := []string{"pull", name}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Debug(":tea: Downloading image " + name)
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Spinner()
|
|
|
|
defer s.ctx.SpinnerStop()
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2019-11-13 08:42:52 +00:00
|
|
|
cmd := exec.Command("img", buildarg...)
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
2020-12-11 22:03:56 +00:00
|
|
|
return errors.Wrap(err, "Failed downloading image: "+string(out))
|
2019-11-13 08:42:52 +00:00
|
|
|
}
|
2019-12-02 15:25:01 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Info(":tea: Image " + name + " downloaded")
|
2021-03-16 13:46:28 +00:00
|
|
|
bus.Manager.Publish(bus.EventImagePostPull, opts)
|
2019-12-02 15:25:01 +00:00
|
|
|
|
2019-11-13 08:42:52 +00:00
|
|
|
return nil
|
|
|
|
}
|
2021-10-20 22:13:02 +00:00
|
|
|
func (s *SimpleImg) CopyImage(src, dst string) error {
|
|
|
|
s.ctx.Spinner()
|
|
|
|
defer s.ctx.SpinnerStop()
|
2019-11-13 08:42:52 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Debug(":tea: Tagging image", src, dst)
|
2019-11-13 08:42:52 +00:00
|
|
|
cmd := exec.Command("img", "tag", src, dst)
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Failed tagging image: "+string(out))
|
|
|
|
}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Info(":tea: Image " + dst + " tagged")
|
2019-12-02 15:25:01 +00:00
|
|
|
|
2019-11-13 08:42:52 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
func (s *SimpleImg) ImageAvailable(imagename string) bool {
|
2020-12-14 17:32:32 +00:00
|
|
|
return imageAvailable(imagename)
|
|
|
|
}
|
|
|
|
|
2021-01-23 21:01:29 +00:00
|
|
|
// ImageExists check if the given image is available locally
|
2020-06-03 19:00:30 +00:00
|
|
|
func (*SimpleImg) ImageExists(imagename string) bool {
|
2021-01-23 21:01:29 +00:00
|
|
|
cmd := exec.Command("img", "ls")
|
|
|
|
out, err := cmd.Output()
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if strings.Contains(string(out), imagename) {
|
|
|
|
return true
|
|
|
|
}
|
2020-06-03 19:00:30 +00:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-04-12 17:00:36 +00:00
|
|
|
func (s *SimpleImg) ImageDefinitionToTar(opts Options) error {
|
2019-11-08 17:29:51 +00:00
|
|
|
if err := s.BuildImage(opts); err != nil {
|
|
|
|
return errors.Wrap(err, "Failed building image")
|
|
|
|
}
|
|
|
|
if err := s.ExportImage(opts); err != nil {
|
|
|
|
return errors.Wrap(err, "Failed exporting image")
|
|
|
|
}
|
|
|
|
if err := s.RemoveImage(opts); err != nil {
|
|
|
|
return errors.Wrap(err, "Failed removing image")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-05 16:37:21 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
func (s *SimpleImg) ExportImage(opts Options) error {
|
2019-11-08 17:29:51 +00:00
|
|
|
name := opts.ImageName
|
|
|
|
path := opts.Destination
|
2019-12-02 15:25:01 +00:00
|
|
|
buildarg := []string{"save", "-o", path, name}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Debug(":tea: Saving image " + name)
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Spinner()
|
|
|
|
defer s.ctx.SpinnerStop()
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2019-11-08 17:29:51 +00:00
|
|
|
out, err := exec.Command("img", buildarg...).CombinedOutput()
|
|
|
|
if err != nil {
|
2020-12-11 22:03:56 +00:00
|
|
|
return errors.Wrap(err, "Failed exporting image: "+string(out))
|
2019-11-08 17:29:51 +00:00
|
|
|
}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Info(":tea: Image " + name + " saved")
|
2019-11-08 17:29:51 +00:00
|
|
|
return nil
|
2019-11-05 16:37:21 +00:00
|
|
|
}
|
|
|
|
|
2021-01-18 11:08:47 +00:00
|
|
|
// ExtractRootfs extracts the docker image content inside the destination
|
2021-04-12 17:00:36 +00:00
|
|
|
func (s *SimpleImg) ExtractRootfs(opts Options, keepPerms bool) error {
|
2019-12-02 15:25:01 +00:00
|
|
|
name := opts.ImageName
|
|
|
|
path := opts.Destination
|
|
|
|
|
2021-01-24 12:17:11 +00:00
|
|
|
if !s.ImageExists(name) {
|
|
|
|
if err := s.DownloadImage(opts); err != nil {
|
|
|
|
return errors.Wrap(err, "failed pulling image "+name+" during extraction")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-02 15:25:01 +00:00
|
|
|
os.RemoveAll(path)
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2019-12-02 15:25:01 +00:00
|
|
|
buildarg := []string{"unpack", "-o", path, name}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Debug(":tea: Extracting image " + name)
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Spinner()
|
|
|
|
defer s.ctx.SpinnerStop()
|
2021-02-13 08:28:54 +00:00
|
|
|
|
2019-12-02 15:25:01 +00:00
|
|
|
out, err := exec.Command("img", buildarg...).CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Failed extracting image: "+string(out))
|
|
|
|
}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Debug(":tea: Image " + name + " extracted")
|
2019-12-02 15:25:01 +00:00
|
|
|
return nil
|
2019-11-10 09:47:28 +00:00
|
|
|
}
|
2019-11-09 12:58:15 +00:00
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
func (s *SimpleImg) Push(opts Options) error {
|
2020-02-15 13:45:05 +00:00
|
|
|
name := opts.ImageName
|
2021-03-16 13:46:28 +00:00
|
|
|
bus.Manager.Publish(bus.EventImagePrePush, opts)
|
|
|
|
|
2020-02-15 13:45:05 +00:00
|
|
|
pusharg := []string{"push", name}
|
|
|
|
out, err := exec.Command("img", pusharg...).CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Failed pushing image: "+string(out))
|
|
|
|
}
|
2021-10-20 22:13:02 +00:00
|
|
|
s.ctx.Info(":tea: Pushed image:", name)
|
2021-03-16 13:46:28 +00:00
|
|
|
bus.Manager.Publish(bus.EventImagePostPush, opts)
|
|
|
|
|
2021-10-20 22:13:02 +00:00
|
|
|
//s.ctx.Info(string(out))
|
2020-02-15 13:45:05 +00:00
|
|
|
return nil
|
|
|
|
}
|