working commit
This commit is contained in:
@@ -1,6 +1,8 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"mstore/pkg/client"
|
||||||
|
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -74,7 +76,7 @@ type ImageInfoParams struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type ImageInfoResult struct {
|
type ImageInfoResult struct {
|
||||||
ImageInfo ImageDescr `json:"imageInfo"`
|
ImageInfo *client.ImageDescr `json:"imageInfo"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (util *ImageUtil) ImageInfo(cmd *cobra.Command, args []string) {
|
func (util *ImageUtil) ImageInfo(cmd *cobra.Command, args []string) {
|
||||||
|
|||||||
@@ -4,111 +4,18 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/google/go-containerregistry/pkg/authn"
|
"mstore/pkg/client"
|
||||||
"github.com/google/go-containerregistry/pkg/crane"
|
|
||||||
"github.com/google/go-containerregistry/pkg/name"
|
|
||||||
pkg "github.com/google/go-containerregistry/pkg/v1"
|
|
||||||
"github.com/google/go-containerregistry/pkg/v1/remote"
|
|
||||||
"github.com/google/go-containerregistry/pkg/v1/remote/transport"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type ImageDescr struct {
|
|
||||||
ConfigFile *pkg.ConfigFile `json:"configFile,omitempty"`
|
|
||||||
Manifest *pkg.Manifest `json:"manifest,omitempty"`
|
|
||||||
Tags []string `json:"avilableTags,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (util *ImageUtil) imageInfo(params *ImageInfoParams) (*ImageInfoResult, error) {
|
func (util *ImageUtil) imageInfo(params *ImageInfoParams) (*ImageInfoResult, error) {
|
||||||
var err error
|
var err error
|
||||||
res := &ImageInfoResult{}
|
res := &ImageInfoResult{}
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
ctx, _ := context.WithTimeout(context.Background(), time.Duration(params.Timeout)*time.Second)
|
cli := client.NewClientWithAuth(params.Username, params.Password)
|
||||||
|
timeout := time.Duration(params.Timeout) * time.Second
|
||||||
options := make([]crane.Option, 0)
|
opres, err := cli.ImageInfo(ctx, params.Imagepath, timeout)
|
||||||
options = append(options, crane.WithContext(ctx))
|
|
||||||
|
|
||||||
ref, err := name.ParseReference(params.Imagepath)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
repo := ref.Context()
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if params.Username != "" && params.Password != "" {
|
|
||||||
defaultTransport := &roundTripper{}
|
|
||||||
scopes := []string{repo.Scope(transport.PullScope)}
|
|
||||||
|
|
||||||
regName := repo.RegistryStr()
|
|
||||||
reg, err := name.NewRegistry(regName)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
basicAuth := &authn.Basic{
|
|
||||||
Username: params.Username,
|
|
||||||
Password: params.Password,
|
|
||||||
}
|
|
||||||
authTransport, err := transport.NewWithContext(ctx, reg, basicAuth, defaultTransport, scopes)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
options = append(options, crane.WithTransport(authTransport))
|
|
||||||
} else {
|
|
||||||
transport := &roundTripper{}
|
|
||||||
options = append(options, crane.WithTransport(transport))
|
|
||||||
}
|
|
||||||
|
|
||||||
image, err := crane.Pull(params.Imagepath, options...)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
|
|
||||||
config, err := image.ConfigFile()
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
res.ImageInfo.ConfigFile = config
|
|
||||||
|
|
||||||
manifest, err := image.Manifest()
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
res.ImageInfo.Manifest = manifest
|
|
||||||
|
|
||||||
return res, err
|
|
||||||
|
|
||||||
remoteOptions := make([]remote.Option, 0)
|
|
||||||
remoteOptions = append(remoteOptions, remote.WithContext(ctx))
|
|
||||||
|
|
||||||
if params.Username != "" && params.Password != "" {
|
|
||||||
defaultTransport := &roundTripper{}
|
|
||||||
scopes := []string{repo.Scope(transport.PullScope)}
|
|
||||||
|
|
||||||
regName := repo.RegistryStr()
|
|
||||||
reg, err := name.NewRegistry(regName)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
basicAuth := &authn.Basic{
|
|
||||||
Username: params.Username,
|
|
||||||
Password: params.Password,
|
|
||||||
}
|
|
||||||
authTransport, err := transport.NewWithContext(ctx, reg, basicAuth, defaultTransport, scopes)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
remoteOptions = append(remoteOptions, remote.WithTransport(authTransport))
|
|
||||||
} else {
|
|
||||||
transport := &roundTripper{}
|
|
||||||
options = append(options, crane.WithTransport(transport))
|
|
||||||
}
|
|
||||||
|
|
||||||
tags, err := remote.List(repo, remoteOptions...)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
res.ImageInfo.Tags = tags
|
|
||||||
|
|
||||||
|
res.ImageInfo = opres
|
||||||
return res, err
|
return res, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,75 +2,20 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"os"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/google/go-containerregistry/pkg/authn"
|
"mstore/pkg/client"
|
||||||
"github.com/google/go-containerregistry/pkg/crane"
|
|
||||||
"github.com/google/go-containerregistry/pkg/name"
|
|
||||||
"github.com/google/go-containerregistry/pkg/v1/remote/transport"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func (util *ImageUtil) pullImage(params *PullImageParams) (*PullImageResult, error) {
|
func (util *ImageUtil) pullImage(params *PullImageParams) (*PullImageResult, error) {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
ctx, _ := context.WithTimeout(context.Background(), time.Duration(params.Timeout)*time.Second)
|
ctx := context.Background()
|
||||||
|
|
||||||
res := &PullImageResult{}
|
res := &PullImageResult{}
|
||||||
options := make([]crane.Option, 0)
|
|
||||||
options = append(options, crane.WithContext(ctx))
|
|
||||||
if params.Username != "" && params.Password != "" {
|
|
||||||
ref, err := name.ParseReference(params.Imagepath)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
repo := ref.Context()
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
defaultTransport := &roundTripper{}
|
|
||||||
|
|
||||||
scopes := []string{repo.Scope(transport.PullScope)}
|
cli := client.NewClientWithAuth(params.Username, params.Password)
|
||||||
|
timeout := time.Duration(params.Timeout) * time.Second
|
||||||
|
err = cli.PullImage(ctx, params.Imagepath, params.Filepath, timeout)
|
||||||
|
|
||||||
regName := repo.RegistryStr()
|
|
||||||
reg, err := name.NewRegistry(regName)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
basicAuth := &authn.Basic{
|
|
||||||
Username: params.Username,
|
|
||||||
Password: params.Password,
|
|
||||||
}
|
|
||||||
authTransport, err := transport.NewWithContext(ctx, reg, basicAuth, defaultTransport, scopes)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
options = append(options, crane.WithTransport(authTransport))
|
|
||||||
} else {
|
|
||||||
transport := &roundTripper{}
|
|
||||||
options = append(options, crane.WithTransport(transport))
|
|
||||||
}
|
|
||||||
|
|
||||||
image, err := crane.Pull(params.Imagepath, options...)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
dstDir := makeTmpFileName(params.Filepath)
|
|
||||||
err = os.MkdirAll(dstDir, 0750)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
err = crane.SaveOCI(image, dstDir)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
err = archiveDir(dstDir, params.Filepath)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
err = os.RemoveAll(dstDir)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
return res, err
|
return res, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,105 +2,19 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/google/go-containerregistry/pkg/authn"
|
"mstore/pkg/client"
|
||||||
"github.com/google/go-containerregistry/pkg/crane"
|
|
||||||
"github.com/google/go-containerregistry/pkg/name"
|
|
||||||
pkg "github.com/google/go-containerregistry/pkg/v1"
|
|
||||||
"github.com/google/go-containerregistry/pkg/v1/layout"
|
|
||||||
"github.com/google/go-containerregistry/pkg/v1/remote/transport"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func (util *ImageUtil) pushImage(params *PushImageParams) (*PushImageResult, error) {
|
func (util *ImageUtil) pushImage(params *PushImageParams) (*PushImageResult, error) {
|
||||||
var err error
|
var err error
|
||||||
ctx, _ := context.WithTimeout(context.Background(), time.Duration(params.Timeout)*time.Second)
|
ctx := context.Background()
|
||||||
res := &PushImageResult{}
|
res := &PushImageResult{}
|
||||||
|
|
||||||
options := make([]crane.Option, 0)
|
cli := client.NewClientWithAuth(params.Username, params.Password)
|
||||||
options = append(options, crane.WithContext(ctx))
|
timeout := time.Duration(params.Timeout) * time.Second
|
||||||
if params.Username != "" && params.Password != "" {
|
err = cli.PushImage(ctx, params.Filepath, params.Imagepath, timeout)
|
||||||
ref, err := name.ParseReference(params.Imagepath)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
repo := ref.Context()
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
defaultTransport := &roundTripper{}
|
|
||||||
|
|
||||||
scopes := []string{
|
|
||||||
repo.Scope(transport.PushScope),
|
|
||||||
repo.Scope(transport.PullScope),
|
|
||||||
}
|
|
||||||
|
|
||||||
regName := repo.RegistryStr()
|
|
||||||
reg, err := name.NewRegistry(regName)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
basicAuth := &authn.Basic{
|
|
||||||
Username: params.Username,
|
|
||||||
Password: params.Password,
|
|
||||||
}
|
|
||||||
|
|
||||||
authTransport, err := transport.NewWithContext(ctx, reg, basicAuth,
|
|
||||||
defaultTransport, scopes)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
options = append(options, crane.WithTransport(authTransport))
|
|
||||||
} else {
|
|
||||||
defaultTransport := &roundTripper{}
|
|
||||||
options = append(options, crane.WithTransport(defaultTransport))
|
|
||||||
}
|
|
||||||
|
|
||||||
dstDir := makeTmpFileName(params.Filepath)
|
|
||||||
err = unarchive(params.Filepath, dstDir)
|
|
||||||
if err != nil {
|
|
||||||
os.RemoveAll(dstDir)
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
image, err := imageLoader(dstDir)
|
|
||||||
if err != nil {
|
|
||||||
os.RemoveAll(dstDir)
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
err = crane.Push(image, params.Imagepath, options...)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func imageLoader(dirPath string) (pkg.Image, error) {
|
|
||||||
var err error
|
|
||||||
var res pkg.Image
|
|
||||||
layoutPath, err := layout.FromPath(dirPath)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
imageIndex, err := layoutPath.ImageIndex()
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
indexManifest, err := imageIndex.IndexManifest()
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
if indexManifest == nil {
|
|
||||||
err := fmt.Errorf("Empty indexManifest referency")
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
|
|
||||||
manifest := indexManifest.Manifests[0]
|
|
||||||
image, err := layoutPath.Image(manifest.Digest)
|
|
||||||
if err != nil {
|
|
||||||
return res, err
|
|
||||||
}
|
|
||||||
res = image
|
|
||||||
return res, err
|
return res, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,14 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/hex"
|
||||||
|
"fmt"
|
||||||
|
"math/rand"
|
||||||
|
)
|
||||||
|
|
||||||
|
func makeTmpFileName(prefix string) string {
|
||||||
|
randBytes := make([]byte, 6)
|
||||||
|
rand.Read(randBytes)
|
||||||
|
suffix := hex.EncodeToString(randBytes)
|
||||||
|
return fmt.Sprintf("%s.tmp.%s", prefix, suffix)
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
package main
|
package client
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
@@ -0,0 +1,75 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"os"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/go-containerregistry/pkg/authn"
|
||||||
|
"github.com/google/go-containerregistry/pkg/crane"
|
||||||
|
"github.com/google/go-containerregistry/pkg/name"
|
||||||
|
"github.com/google/go-containerregistry/pkg/v1/remote/transport"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (cli *Client) PullImage(ctx context.Context, filepath, imagepath string, timeout time.Duration) error {
|
||||||
|
var err error
|
||||||
|
ctx, _ = context.WithTimeout(ctx, timeout)
|
||||||
|
|
||||||
|
options := make([]crane.Option, 0)
|
||||||
|
options = append(options, crane.WithContext(ctx))
|
||||||
|
|
||||||
|
if cli.username != "" && cli.password != "" {
|
||||||
|
ref, err := name.ParseReference(imagepath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
repo := ref.Context()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defaultTransport := &roundTripper{}
|
||||||
|
|
||||||
|
scopes := []string{repo.Scope(transport.PullScope)}
|
||||||
|
|
||||||
|
regName := repo.RegistryStr()
|
||||||
|
reg, err := name.NewRegistry(regName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
basicAuth := &authn.Basic{
|
||||||
|
Username: cli.username,
|
||||||
|
Password: cli.password,
|
||||||
|
}
|
||||||
|
authTransport, err := transport.NewWithContext(ctx, reg, basicAuth, defaultTransport, scopes)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
options = append(options, crane.WithTransport(authTransport))
|
||||||
|
} else {
|
||||||
|
transport := &roundTripper{}
|
||||||
|
options = append(options, crane.WithTransport(transport))
|
||||||
|
}
|
||||||
|
|
||||||
|
image, err := crane.Pull(imagepath, options...)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
dstdir := makeTmpFileName(filepath)
|
||||||
|
err = os.MkdirAll(dstdir, 0750)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = crane.SaveOCI(image, dstdir)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = archiveDir(dstdir, filepath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = os.RemoveAll(dstdir)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
@@ -0,0 +1,104 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/go-containerregistry/pkg/authn"
|
||||||
|
"github.com/google/go-containerregistry/pkg/crane"
|
||||||
|
"github.com/google/go-containerregistry/pkg/name"
|
||||||
|
pkg "github.com/google/go-containerregistry/pkg/v1"
|
||||||
|
"github.com/google/go-containerregistry/pkg/v1/layout"
|
||||||
|
"github.com/google/go-containerregistry/pkg/v1/remote/transport"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (cli *Client) PushImage(ctx context.Context, filepath, imagepath string, timeout time.Duration) error {
|
||||||
|
var err error
|
||||||
|
ctx, _ = context.WithTimeout(ctx, timeout)
|
||||||
|
|
||||||
|
options := make([]crane.Option, 0)
|
||||||
|
options = append(options, crane.WithContext(ctx))
|
||||||
|
if cli.username != "" && cli.password != "" {
|
||||||
|
ref, err := name.ParseReference(imagepath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
repo := ref.Context()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defaultTransport := &roundTripper{}
|
||||||
|
|
||||||
|
scopes := []string{
|
||||||
|
repo.Scope(transport.PushScope),
|
||||||
|
repo.Scope(transport.PullScope),
|
||||||
|
}
|
||||||
|
|
||||||
|
regName := repo.RegistryStr()
|
||||||
|
reg, err := name.NewRegistry(regName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
basicAuth := &authn.Basic{
|
||||||
|
Username: cli.username,
|
||||||
|
Password: cli.password,
|
||||||
|
}
|
||||||
|
|
||||||
|
authTransport, err := transport.NewWithContext(ctx, reg, basicAuth, defaultTransport, scopes)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
options = append(options, crane.WithTransport(authTransport))
|
||||||
|
} else {
|
||||||
|
defaultTransport := &roundTripper{}
|
||||||
|
options = append(options, crane.WithTransport(defaultTransport))
|
||||||
|
}
|
||||||
|
|
||||||
|
dstdir := makeTmpFileName(filepath)
|
||||||
|
err = unarchive(filepath, dstdir)
|
||||||
|
if err != nil {
|
||||||
|
os.RemoveAll(dstdir)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
image, err := imageLoader(dstdir)
|
||||||
|
if err != nil {
|
||||||
|
os.RemoveAll(dstdir)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = crane.Push(image, imagepath, options...)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func imageLoader(dirPath string) (pkg.Image, error) {
|
||||||
|
var err error
|
||||||
|
var res pkg.Image
|
||||||
|
layoutPath, err := layout.FromPath(dirPath)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
imageIndex, err := layoutPath.ImageIndex()
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
indexManifest, err := imageIndex.IndexManifest()
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
if indexManifest == nil {
|
||||||
|
err := fmt.Errorf("Empty indexManifest referency")
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
manifest := indexManifest.Manifests[0]
|
||||||
|
image, err := layoutPath.Image(manifest.Digest)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
res = image
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
@@ -1,11 +1,8 @@
|
|||||||
package main
|
package client
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"archive/tar"
|
"archive/tar"
|
||||||
"encoding/hex"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"math/rand"
|
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
@@ -78,7 +75,6 @@ func unarchive(filePath, dstDir string) error {
|
|||||||
defer file.Close()
|
defer file.Close()
|
||||||
|
|
||||||
tarReader := tar.NewReader(file)
|
tarReader := tar.NewReader(file)
|
||||||
//defer tarReader.Close()
|
|
||||||
for {
|
for {
|
||||||
header, err := tarReader.Next()
|
header, err := tarReader.Next()
|
||||||
switch {
|
switch {
|
||||||
@@ -120,10 +116,3 @@ func unarchive(filePath, dstDir string) error {
|
|||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func makeTmpFileName(prefix string) string {
|
|
||||||
randBytes := make([]byte, 6)
|
|
||||||
rand.Read(randBytes)
|
|
||||||
suffix := hex.EncodeToString(randBytes)
|
|
||||||
return fmt.Sprintf("%s.tmp.%s", prefix, suffix)
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user