wisski-cloud-distillery/internal/wisski/ingredient/trb/trb.go
Tom Wiesing 8235ea9105
Require access to Still via method
This commit adds a safeguard to accessing the still from a specific
component by requiring access via the component.GetStill method.
2024-04-08 22:39:32 +02:00

119 lines
2.8 KiB
Go

package trb
import (
"compress/gzip"
"context"
"errors"
"fmt"
"io"
"os"
"github.com/FAU-CDI/wisski-distillery/internal/wisski/ingredient"
"github.com/FAU-CDI/wisski-distillery/internal/wisski/ingredient/barrel"
"github.com/FAU-CDI/wisski-distillery/pkg/logging"
)
type TRB struct {
ingredient.Base
dependencies struct {
Barrel *barrel.Barrel
}
}
// RebuildTriplestore rebuilds the triplestore by making a backup, storing it on disk, purging the triplestore, and restoring the backup.
func (trb *TRB) RebuildTriplestore(ctx context.Context, out io.Writer, allowEmptyRepository bool) (err error) {
// stop instance, restart when done
logging.LogMessage(out, "Shutting down instance")
if err := trb.dependencies.Barrel.Stack().Down(ctx, out); err != nil {
return err
}
defer func() {
logging.LogMessage(out, "Restarting instance")
e := trb.dependencies.Barrel.Stack().Up(ctx, out)
if err == nil {
err = e
}
}()
// make the backup
logging.LogMessage(out, "Storing triplestore content")
dumpPath, err := trb.makeBackup(ctx, allowEmptyRepository)
if err != nil {
return err
}
fmt.Printf("Wrote %q\n", dumpPath)
liquid := ingredient.GetLiquid(trb)
logging.LogMessage(out, "Purging triplestore")
if err := liquid.TS.Purge(ctx, liquid.Instance, liquid.Domain()); err != nil {
return err
}
logging.LogMessage(out, "Provising triplestore")
if err := liquid.TS.Provision(ctx, liquid.Instance, liquid.Domain()); err != nil {
return err
}
logging.LogMessage(out, "Restoring triplestore")
if err := trb.restoreBackup(ctx, dumpPath); err != nil {
return err
}
logging.LogMessage(out, "Deleting dump file")
if err := os.Remove(dumpPath); err != nil {
return err
}
return
}
var errBackupEmpty = errors.New("no data contained in backup file (is the repository empty?)")
func (trb *TRB) makeBackup(ctx context.Context, allowEmptyRepository bool) (path string, err error) {
file, err := os.CreateTemp("", "*.nq.gz")
if err != nil {
return "", err
}
defer file.Close()
// create a new writer
zippedFile := gzip.NewWriter(file)
defer zippedFile.Close()
liquid := ingredient.GetLiquid(trb)
count, err := liquid.TS.SnapshotDB(ctx, zippedFile, liquid.GraphDBRepository)
if err != nil {
return "", err
}
if count == 0 && !allowEmptyRepository {
return "", errBackupEmpty
}
return file.Name(), nil
}
func (trb *TRB) restoreBackup(ctx context.Context, path string) (err error) {
reader, err := os.Open(path)
if err != nil {
return err
}
defer reader.Close()
decompressedReader, err := gzip.NewReader(reader)
if err != nil {
return err
}
defer decompressedReader.Close()
liquid := ingredient.GetLiquid(trb)
if err := liquid.TS.RestoreDB(ctx, liquid.GraphDBRepository, decompressedReader); err != nil {
return err
}
return nil
}