- Add SQLite database connection management with proper error handling - Implement schema for files, chunks, blobs, and snapshots tables - Create repository pattern for each database table - Add transaction support with proper rollback handling - Integrate database module with fx dependency injection - Make index path configurable via VAULTIK_INDEX_PATH env var - Add fatal error handling for database integrity issues - Update DESIGN.md to clarify file_chunks vs chunk_files distinction - Remove FinalHash from BlobInfo (blobs are content-addressable) - Add file metadata support (mtime, ctime, mode, uid, gid, symlinks)
89 lines
2.2 KiB
Go
89 lines
2.2 KiB
Go
package cli
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
|
|
"git.eeqj.de/sneak/vaultik/internal/globals"
|
|
"github.com/spf13/cobra"
|
|
"go.uber.org/fx"
|
|
)
|
|
|
|
// FetchOptions contains options for the fetch command
|
|
type FetchOptions struct {
|
|
Bucket string
|
|
Prefix string
|
|
SnapshotID string
|
|
FilePath string
|
|
Target string
|
|
}
|
|
|
|
// NewFetchCommand creates the fetch command
|
|
func NewFetchCommand() *cobra.Command {
|
|
opts := &FetchOptions{}
|
|
|
|
cmd := &cobra.Command{
|
|
Use: "fetch",
|
|
Short: "Extract single file from backup",
|
|
Long: `Download and decrypt a single file from a backup snapshot`,
|
|
Args: cobra.NoArgs,
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
// Validate required flags
|
|
if opts.Bucket == "" {
|
|
return fmt.Errorf("--bucket is required")
|
|
}
|
|
if opts.Prefix == "" {
|
|
return fmt.Errorf("--prefix is required")
|
|
}
|
|
if opts.SnapshotID == "" {
|
|
return fmt.Errorf("--snapshot is required")
|
|
}
|
|
if opts.FilePath == "" {
|
|
return fmt.Errorf("--file is required")
|
|
}
|
|
if opts.Target == "" {
|
|
return fmt.Errorf("--target is required")
|
|
}
|
|
return runFetch(cmd.Context(), opts)
|
|
},
|
|
}
|
|
|
|
cmd.Flags().StringVar(&opts.Bucket, "bucket", "", "S3 bucket name")
|
|
cmd.Flags().StringVar(&opts.Prefix, "prefix", "", "S3 prefix")
|
|
cmd.Flags().StringVar(&opts.SnapshotID, "snapshot", "", "Snapshot ID")
|
|
cmd.Flags().StringVar(&opts.FilePath, "file", "", "Path of file to extract from backup")
|
|
cmd.Flags().StringVar(&opts.Target, "target", "", "Target path for extracted file")
|
|
|
|
return cmd
|
|
}
|
|
|
|
func runFetch(ctx context.Context, opts *FetchOptions) error {
|
|
if os.Getenv("VAULTIK_PRIVATE_KEY") == "" {
|
|
return fmt.Errorf("VAULTIK_PRIVATE_KEY environment variable must be set")
|
|
}
|
|
|
|
app := fx.New(
|
|
fx.Supply(opts),
|
|
fx.Provide(globals.New),
|
|
// Additional modules will be added here
|
|
fx.Invoke(func(g *globals.Globals) error {
|
|
// TODO: Implement fetch logic
|
|
fmt.Printf("Fetching %s from snapshot %s to %s\n", opts.FilePath, opts.SnapshotID, opts.Target)
|
|
return nil
|
|
}),
|
|
fx.NopLogger,
|
|
)
|
|
|
|
if err := app.Start(ctx); err != nil {
|
|
return fmt.Errorf("failed to start fetch: %w", err)
|
|
}
|
|
defer func() {
|
|
if err := app.Stop(ctx); err != nil {
|
|
fmt.Printf("error stopping app: %v\n", err)
|
|
}
|
|
}()
|
|
|
|
return nil
|
|
}
|