Files
steampipe/pkg/ociinstaller/plugin.go
kaidaguerre 404dd35e21 Update database code to use pgx interface so we can leverage the connection pool hook functions to pre-warm connections. Closes #2422 (#2438)
* Provide feedback for failed prepared statements
* Move error functions to error_helpers
* Make maintenance client retriable
2022-10-05 12:38:57 +01:00

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
}