2024-02-05 19:44:47 +00:00
|
|
|
package plugin
|
|
|
|
|
|
|
|
import (
|
|
|
|
"archive/zip"
|
|
|
|
"context"
|
2024-02-07 12:17:58 +00:00
|
|
|
"errors"
|
2024-02-05 19:44:47 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/Masterminds/semver/v3"
|
2024-02-07 12:17:58 +00:00
|
|
|
"github.com/rs/zerolog/log"
|
2024-05-12 10:03:10 +00:00
|
|
|
"github.com/thegeeklab/wp-opentofu/tofu"
|
2024-02-05 19:44:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func installPackage(ctx context.Context, client *http.Client, version string, maxSize int64) error {
|
|
|
|
// Sanitize user input
|
2024-02-07 12:17:58 +00:00
|
|
|
semverVersion, err := semver.NewVersion(version)
|
|
|
|
if err != nil {
|
2024-02-05 20:50:41 +00:00
|
|
|
return fmt.Errorf("%w: %v", ErrInvalidTofuVersion, version)
|
2024-02-05 19:44:47 +00:00
|
|
|
}
|
|
|
|
|
2024-02-07 12:17:58 +00:00
|
|
|
packageURL := fmt.Sprintf(
|
|
|
|
"https://github.com/opentofu/opentofu/releases/download/v%s/tofu_%s_linux_amd64.zip",
|
|
|
|
semverVersion.String(),
|
|
|
|
semverVersion.String(),
|
2024-02-05 19:44:47 +00:00
|
|
|
)
|
2024-02-07 12:17:58 +00:00
|
|
|
|
|
|
|
tmpdir, err := os.MkdirTemp("/tmp", "tofu_dl_")
|
2024-02-05 19:44:47 +00:00
|
|
|
if err != nil {
|
2024-02-07 12:17:58 +00:00
|
|
|
return fmt.Errorf("failed to create tmp dir: %w", err)
|
2024-02-05 19:44:47 +00:00
|
|
|
}
|
|
|
|
|
2024-03-14 19:45:00 +00:00
|
|
|
defer func() {
|
2024-05-12 10:03:10 +00:00
|
|
|
_ = os.RemoveAll(tmpdir)
|
2024-03-14 19:45:00 +00:00
|
|
|
}()
|
2024-02-07 12:17:58 +00:00
|
|
|
|
|
|
|
log.Debug().
|
|
|
|
Str("tmpdir", tmpdir).
|
|
|
|
Msgf("Download OpenTofu '%s' from URL '%s'", version, packageURL)
|
|
|
|
|
|
|
|
tmpfile := filepath.Join(tmpdir, "tofu.zip")
|
|
|
|
|
|
|
|
if err := downloadPackage(ctx, client, tmpfile, packageURL); err != nil {
|
|
|
|
return fmt.Errorf("failed to download: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := unzip(tmpfile, tmpdir, maxSize); err != nil {
|
|
|
|
return fmt.Errorf("failed to unzip: %w", err)
|
|
|
|
}
|
|
|
|
|
2024-05-12 10:03:10 +00:00
|
|
|
if err := os.Rename(filepath.Join(tmpdir, "tofu"), tofu.TofuBin); err != nil {
|
2024-02-07 12:17:58 +00:00
|
|
|
return fmt.Errorf("failed to rename: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2024-02-05 19:44:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func downloadPackage(ctx context.Context, client *http.Client, filepath, url string) error {
|
|
|
|
// Create the file
|
|
|
|
out, err := os.Create(filepath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer out.Close()
|
|
|
|
|
|
|
|
// Get the data
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-02-07 12:17:58 +00:00
|
|
|
|
|
|
|
if resp.StatusCode > http.StatusBadRequest {
|
|
|
|
return fmt.Errorf("%w: %v", ErrHTTPError, resp.Status)
|
|
|
|
}
|
|
|
|
|
2024-02-05 19:44:47 +00:00
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
// Writer the body to file
|
|
|
|
_, err = io.Copy(out, resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func unzip(src, dest string, maxSize int64) error {
|
|
|
|
r, err := zip.OpenReader(src)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if err := r.Close(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
_ = os.MkdirAll(dest, defaultDirPerm)
|
|
|
|
|
|
|
|
// Closure to address file descriptors issue with all the deferred .Close() methods
|
|
|
|
extractAndWriteFile := func(f *zip.File) error {
|
|
|
|
rc, err := f.Open()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if err := rc.Close(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
path, err := sanitizeArchivePath(dest, f.Name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.FileInfo().IsDir() { //nolint: nestif
|
|
|
|
_ = os.MkdirAll(path, f.Mode())
|
|
|
|
} else {
|
|
|
|
_ = os.MkdirAll(filepath.Dir(path), f.Mode())
|
2024-02-12 08:25:46 +00:00
|
|
|
|
2024-02-05 19:44:47 +00:00
|
|
|
f, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, f.Mode())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if err := f.Close(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
written, err := io.CopyN(f, rc, maxSize)
|
2024-02-07 12:17:58 +00:00
|
|
|
if err != nil && !errors.Is(err, io.EOF) {
|
2024-02-05 19:44:47 +00:00
|
|
|
return err
|
|
|
|
} else if written == maxSize {
|
|
|
|
return fmt.Errorf("%w: %d", ErrMaxSizeSizeLimit, maxSize)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, f := range r.File {
|
|
|
|
err := extractAndWriteFile(f)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func sanitizeArchivePath(d, t string) (string, error) {
|
|
|
|
value := filepath.Join(d, t)
|
|
|
|
if strings.HasPrefix(value, filepath.Clean(d)) {
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", fmt.Errorf("%w: %v", ErrTaintedPath, t)
|
|
|
|
}
|