2019-11-10 09:48:07 +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 helpers
|
|
|
|
|
|
|
|
import (
|
2020-02-01 16:58:23 +00:00
|
|
|
"archive/tar"
|
2020-06-06 10:34:44 +00:00
|
|
|
"bytes"
|
2019-11-10 09:48:07 +00:00
|
|
|
"io"
|
|
|
|
"os"
|
2020-02-01 16:58:23 +00:00
|
|
|
"path/filepath"
|
|
|
|
|
2019-11-10 09:48:07 +00:00
|
|
|
"github.com/docker/docker/pkg/archive"
|
|
|
|
)
|
|
|
|
|
|
|
|
func Tar(src, dest string) error {
|
|
|
|
out, err := os.Create(dest)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer out.Close()
|
|
|
|
|
|
|
|
fs, err := archive.Tar(src, archive.Uncompressed)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer fs.Close()
|
|
|
|
|
|
|
|
_, err = io.Copy(out, fs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = out.Sync()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-06-02 07:04:40 +00:00
|
|
|
type TarModifierWrapperFunc func(path, dst string, header *tar.Header, content io.Reader) (*tar.Header, []byte, error)
|
|
|
|
type TarModifierWrapper struct {
|
|
|
|
DestinationPath string
|
|
|
|
Modifier TarModifierWrapperFunc
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewTarModifierWrapper(dst string, modifier TarModifierWrapperFunc) *TarModifierWrapper {
|
|
|
|
return &TarModifierWrapper{
|
|
|
|
DestinationPath: dst,
|
|
|
|
Modifier: modifier,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *TarModifierWrapper) GetModifier() archive.TarModifierFunc {
|
|
|
|
return func(path string, header *tar.Header, content io.Reader) (*tar.Header, []byte, error) {
|
|
|
|
return m.Modifier(m.DestinationPath, path, header, content)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func UntarProtect(src, dst string, sameOwner bool, protectedFiles []string, modifier *TarModifierWrapper) error {
|
|
|
|
var ans error
|
|
|
|
|
|
|
|
if len(protectedFiles) <= 0 {
|
|
|
|
return Untar(src, dst, sameOwner)
|
|
|
|
}
|
|
|
|
|
|
|
|
// POST: we have files to protect. I create a ReplaceFileTarWrapper
|
|
|
|
in, err := os.Open(src)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer in.Close()
|
|
|
|
|
|
|
|
// Create modifier map
|
|
|
|
mods := make(map[string]archive.TarModifierFunc)
|
|
|
|
for _, file := range protectedFiles {
|
|
|
|
mods[file] = modifier.GetModifier()
|
|
|
|
}
|
|
|
|
|
|
|
|
if sameOwner {
|
|
|
|
// PRE: i have root privileged.
|
|
|
|
|
2020-06-06 10:34:44 +00:00
|
|
|
replacerArchive := archive.ReplaceFileTarWrapper(in, mods)
|
|
|
|
|
2020-06-02 07:04:40 +00:00
|
|
|
opts := &archive.TarOptions{
|
|
|
|
// NOTE: NoLchown boolean is used for chmod of the symlink
|
|
|
|
// Probably it's needed set this always to true.
|
|
|
|
NoLchown: true,
|
|
|
|
ExcludePatterns: []string{"dev/"}, // prevent 'operation not permitted'
|
|
|
|
ContinueOnError: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
ans = archive.Untar(replacerArchive, dst, opts)
|
|
|
|
} else {
|
2020-06-06 10:34:44 +00:00
|
|
|
ans = unTarIgnoreOwner(dst, in, mods)
|
2020-06-02 07:04:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ans
|
|
|
|
}
|
|
|
|
|
2020-06-06 10:34:44 +00:00
|
|
|
func unTarIgnoreOwner(dest string, in io.ReadCloser, mods map[string]archive.TarModifierFunc) error {
|
|
|
|
tr := tar.NewReader(in)
|
|
|
|
for {
|
|
|
|
header, err := tr.Next()
|
|
|
|
|
|
|
|
var data []byte
|
|
|
|
var headerReplaced = false
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case err == io.EOF:
|
|
|
|
goto tarEof
|
|
|
|
case err != nil:
|
|
|
|
return err
|
|
|
|
case header == nil:
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// the target location where the dir/file should be created
|
|
|
|
target := filepath.Join(dest, header.Name)
|
|
|
|
if mods != nil {
|
|
|
|
modifier, ok := mods[header.Name]
|
|
|
|
if ok {
|
|
|
|
header, data, err = modifier(header.Name, header, tr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Override target path
|
|
|
|
target = filepath.Join(dest, header.Name)
|
|
|
|
headerReplaced = true
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the file type
|
|
|
|
switch header.Typeflag {
|
|
|
|
|
|
|
|
// if its a dir and it doesn't exist create it
|
|
|
|
case tar.TypeDir:
|
|
|
|
if _, err := os.Stat(target); err != nil {
|
|
|
|
if err := os.MkdirAll(target, 0755); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle creation of file
|
|
|
|
case tar.TypeReg:
|
|
|
|
|
|
|
|
f, err := os.OpenFile(target, os.O_CREATE|os.O_RDWR, os.FileMode(header.Mode))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// copy over contents
|
|
|
|
if headerReplaced {
|
|
|
|
_, err = io.Copy(f, bytes.NewReader(data))
|
|
|
|
} else {
|
|
|
|
_, err = io.Copy(f, tr)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// manually close here after each file operation; defering would cause each
|
|
|
|
// file close to wait until all operations have completed.
|
|
|
|
f.Close()
|
|
|
|
|
|
|
|
case tar.TypeSymlink:
|
|
|
|
source := header.Linkname
|
|
|
|
err := os.Symlink(source, target)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tarEof:
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-10 09:48:07 +00:00
|
|
|
// Untar just a wrapper around the docker functions
|
|
|
|
func Untar(src, dest string, sameOwner bool) error {
|
2020-02-01 16:58:23 +00:00
|
|
|
var ans error
|
|
|
|
|
2019-11-10 09:48:07 +00:00
|
|
|
in, err := os.Open(src)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer in.Close()
|
|
|
|
|
2020-02-11 15:28:10 +00:00
|
|
|
if sameOwner {
|
2020-02-01 16:58:23 +00:00
|
|
|
// PRE: i have root privileged.
|
2020-01-12 22:31:43 +00:00
|
|
|
|
2020-02-01 16:58:23 +00:00
|
|
|
opts := &archive.TarOptions{
|
|
|
|
// NOTE: NoLchown boolean is used for chmod of the symlink
|
|
|
|
// Probably it's needed set this always to true.
|
|
|
|
NoLchown: true,
|
|
|
|
ExcludePatterns: []string{"dev/"}, // prevent 'operation not permitted'
|
2020-04-18 09:41:34 +00:00
|
|
|
ContinueOnError: true,
|
2020-02-01 16:58:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ans = archive.Untar(in, dest, opts)
|
|
|
|
} else {
|
2020-06-06 10:34:44 +00:00
|
|
|
ans = unTarIgnoreOwner(dest, in, nil)
|
2020-01-12 22:31:43 +00:00
|
|
|
}
|
|
|
|
|
2020-02-01 16:58:23 +00:00
|
|
|
return ans
|
2019-11-10 09:48:07 +00:00
|
|
|
}
|