mirror of
https://github.com/turbot/steampipe.git
synced 2026-02-18 04:00:20 -05:00
* Provide feedback for failed prepared statements * Move error functions to error_helpers * Make maintenance client retriable
236 lines
7.0 KiB
Go
236 lines
7.0 KiB
Go
package ociinstaller
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/turbot/steampipe/pkg/error_helpers"
|
|
"github.com/turbot/steampipe/pkg/filepaths"
|
|
versionfile "github.com/turbot/steampipe/pkg/ociinstaller/versionfile"
|
|
"github.com/turbot/steampipe/pkg/utils"
|
|
)
|
|
|
|
// InstallPlugin installs a plugin from an OCI Image
|
|
func InstallPlugin(ctx context.Context, imageRef string, sub chan struct{}) (*SteampipeImage, error) {
|
|
tempDir := NewTempDir(filepaths.EnsurePluginDir())
|
|
defer func() {
|
|
// send a last beacon to signal completion
|
|
sub <- struct{}{}
|
|
if err := tempDir.Delete(); err != nil {
|
|
log.Printf("[TRACE] Failed to delete temp dir '%s' after installing plugin: %s", tempDir, err)
|
|
}
|
|
}()
|
|
|
|
ref := NewSteampipeImageRef(imageRef)
|
|
imageDownloader := NewOciDownloader()
|
|
|
|
sub <- struct{}{}
|
|
image, err := imageDownloader.Download(ctx, ref, ImageTypePlugin, tempDir.Path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
sub <- struct{}{}
|
|
if err = installPluginBinary(image, tempDir.Path); err != nil {
|
|
return nil, fmt.Errorf("plugin installation failed: %s", err)
|
|
}
|
|
sub <- struct{}{}
|
|
if err = installPluginDocs(image, tempDir.Path); err != nil {
|
|
return nil, fmt.Errorf("plugin installation failed: %s", err)
|
|
}
|
|
sub <- struct{}{}
|
|
if err = installPluginConfigFiles(image, tempDir.Path); err != nil {
|
|
return nil, fmt.Errorf("plugin installation failed: %s", err)
|
|
}
|
|
|
|
sub <- struct{}{}
|
|
if err := updateVersionFilePlugin(image); err != nil {
|
|
return nil, err
|
|
}
|
|
return image, nil
|
|
}
|
|
|
|
var versionFileUpdateLock = &sync.Mutex{}
|
|
|
|
func updateVersionFilePlugin(image *SteampipeImage) error {
|
|
versionFileUpdateLock.Lock()
|
|
defer versionFileUpdateLock.Unlock()
|
|
|
|
timeNow := versionfile.FormatTime(time.Now())
|
|
v, err := versionfile.LoadPluginVersionFile()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
pluginFullName := image.ImageRef.DisplayImageRef()
|
|
|
|
plugin, ok := v.Plugins[pluginFullName]
|
|
if !ok {
|
|
plugin = &versionfile.InstalledVersion{}
|
|
}
|
|
|
|
//change this to the path????
|
|
plugin.Name = pluginFullName
|
|
plugin.Version = image.Config.Plugin.Version
|
|
plugin.ImageDigest = string(image.OCIDescriptor.Digest)
|
|
plugin.BinaryDigest = image.Plugin.BinaryDigest
|
|
plugin.BinaryArchitecture = image.Plugin.BinaryArchitecture
|
|
plugin.InstalledFrom = image.ImageRef.ActualImageRef()
|
|
plugin.LastCheckedDate = timeNow
|
|
plugin.InstallDate = timeNow
|
|
|
|
v.Plugins[pluginFullName] = plugin
|
|
|
|
return v.Save()
|
|
}
|
|
|
|
func installPluginBinary(image *SteampipeImage, tempdir string) error {
|
|
sourcePath := filepath.Join(tempdir, image.Plugin.BinaryFile)
|
|
destDir := pluginInstallDir(image.ImageRef)
|
|
|
|
// check if system is M1 - if so we need some special handling
|
|
isM1, err := utils.IsMacM1()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to detect system architecture")
|
|
}
|
|
if isM1 {
|
|
// NOTE: for Mac M1 machines, if the binary is updated in place without deleting the existing file,
|
|
// the updated plugin binary may crash on execution - for an undetermined reason
|
|
// to avoid this, remove the existing plugin folder and re-create it
|
|
if err := os.RemoveAll(destDir); err != nil {
|
|
return fmt.Errorf("could not remove plugin folder")
|
|
}
|
|
if err := os.MkdirAll(destDir, 0755); err != nil {
|
|
return fmt.Errorf("could not create plugin folder")
|
|
}
|
|
}
|
|
|
|
// unzip the file into the plugin folder
|
|
if _, err := ungzip(sourcePath, destDir); err != nil {
|
|
return fmt.Errorf("could not unzip %s to %s", sourcePath, destDir)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func installPluginDocs(image *SteampipeImage, tempdir string) error {
|
|
installTo := pluginInstallDir(image.ImageRef)
|
|
|
|
// if DocsDir is not set, then there are no docs.
|
|
if image.Plugin.DocsDir == "" {
|
|
return nil
|
|
}
|
|
|
|
// install the docs
|
|
sourcePath := filepath.Join(tempdir, image.Plugin.DocsDir)
|
|
destPath := filepath.Join(installTo, "docs")
|
|
if fileExists(destPath) {
|
|
os.RemoveAll(destPath)
|
|
}
|
|
if err := moveFolderWithinPartition(sourcePath, destPath); err != nil {
|
|
return fmt.Errorf("could not copy %s to %s", sourcePath, destPath)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func installPluginConfigFiles(image *SteampipeImage, tempdir string) error {
|
|
installTo := filepaths.EnsureConfigDir()
|
|
|
|
// if ConfigFileDir is not set, then there are no config files.
|
|
if image.Plugin.ConfigFileDir == "" {
|
|
return nil
|
|
}
|
|
// install config files (if they dont already exist)
|
|
sourcePath := filepath.Join(tempdir, image.Plugin.ConfigFileDir)
|
|
directory, err := os.Open(sourcePath)
|
|
if err != nil {
|
|
return fmt.Errorf("couldn't open source dir: %s", err)
|
|
}
|
|
defer directory.Close()
|
|
|
|
objects, err := directory.Readdir(-1)
|
|
if err != nil {
|
|
return fmt.Errorf("couldn't read source dir: %s", err)
|
|
}
|
|
|
|
for _, obj := range objects {
|
|
sourceFile := filepath.Join(sourcePath, obj.Name())
|
|
destFile := filepath.Join(installTo, obj.Name())
|
|
if err := copyConfigFileUnlessExists(sourceFile, destFile, image.ImageRef); err != nil {
|
|
return fmt.Errorf("could not copy config file from %s to %s", sourceFile, destFile)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func copyConfigFileUnlessExists(sourceFile string, destFile string, ref *SteampipeImageRef) error {
|
|
if fileExists(destFile) {
|
|
return nil
|
|
}
|
|
inputData, err := os.ReadFile(sourceFile)
|
|
if err != nil {
|
|
return fmt.Errorf("couldn't open source file: %s", err)
|
|
}
|
|
inputStat, err := os.Stat(sourceFile)
|
|
if err != nil {
|
|
return fmt.Errorf("couldn't read source file permissions: %s", err)
|
|
}
|
|
// update the connection config with the correct plugin version
|
|
inputData = addPluginStreamToConfig(inputData, ref)
|
|
if err = os.WriteFile(destFile, inputData, inputStat.Mode()); err != nil {
|
|
return fmt.Errorf("writing to output file failed: %s", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// The default config files have the plugin set to the 'latest' stream (as this is what is installed by default)
|
|
// When installing non-latest plugins, that property needs to be adjusted to the stream actually getting installed.
|
|
// Otherwise, during plugin resolution, it will resolve to an incorrect plugin instance
|
|
// (or none at at all, if 'latest' versions isn't installed)
|
|
func addPluginStreamToConfig(src []byte, ref *SteampipeImageRef) []byte {
|
|
_, _, stream := ref.GetOrgNameAndStream()
|
|
if stream == "latest" {
|
|
return src
|
|
}
|
|
|
|
regex := regexp.MustCompile(`^(\s*)plugin\s*=\s*"(.*)"\s*$`)
|
|
substitution := fmt.Sprintf(`$1 plugin = "$2@%s"`, stream)
|
|
|
|
srcScanner := bufio.NewScanner(strings.NewReader(string(src)))
|
|
srcScanner.Split(bufio.ScanLines)
|
|
destBuffer := bytes.NewBufferString("")
|
|
|
|
for srcScanner.Scan() {
|
|
line := srcScanner.Text()
|
|
if regex.MatchString(line) {
|
|
line = regex.ReplaceAllString(line, substitution)
|
|
// remove the extra space we had to add to the substitution token
|
|
line = line[1:]
|
|
}
|
|
destBuffer.WriteString(fmt.Sprintf("%s\n", line))
|
|
}
|
|
return destBuffer.Bytes()
|
|
}
|
|
|
|
func pluginInstallDir(ref *SteampipeImageRef) string {
|
|
osSafePath := filepath.FromSlash(ref.DisplayImageRef())
|
|
|
|
fullPath := filepath.Join(filepaths.EnsurePluginDir(), osSafePath)
|
|
|
|
if _, err := os.Stat(fullPath); os.IsNotExist(err) {
|
|
err = os.MkdirAll(fullPath, 0755)
|
|
error_helpers.FailOnErrorWithMessage(err, "could not create plugin install directory")
|
|
}
|
|
|
|
return fullPath
|
|
}
|