2024-03-18 14:41:58 +00:00
|
|
|
package deploy
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"io"
|
|
|
|
"io/fs"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/databricks/cli/bundle"
|
|
|
|
"github.com/databricks/cli/bundle/deploy/files"
|
2024-03-25 14:18:47 +00:00
|
|
|
"github.com/databricks/cli/libs/diag"
|
2024-03-18 14:41:58 +00:00
|
|
|
"github.com/databricks/cli/libs/filer"
|
|
|
|
"github.com/databricks/cli/libs/log"
|
|
|
|
"github.com/databricks/cli/libs/sync"
|
|
|
|
)
|
|
|
|
|
|
|
|
type statePull struct {
|
|
|
|
filerFactory FilerFactory
|
|
|
|
}
|
|
|
|
|
2024-03-25 14:18:47 +00:00
|
|
|
func (s *statePull) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
2024-03-18 14:41:58 +00:00
|
|
|
f, err := s.filerFactory(b)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Download deployment state file from filer to local cache directory.
|
|
|
|
log.Infof(ctx, "Opening remote deployment state file")
|
|
|
|
remote, err := s.remoteState(ctx, f)
|
|
|
|
if err != nil {
|
|
|
|
log.Infof(ctx, "Unable to open remote deployment state file: %s", err)
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
if remote == nil {
|
|
|
|
log.Infof(ctx, "Remote deployment state file does not exist")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
statePath, err := getPathToStateFile(ctx, b)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
local, err := os.OpenFile(statePath, os.O_CREATE|os.O_RDWR, 0o600)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
defer local.Close()
|
|
|
|
|
|
|
|
data := remote.Bytes()
|
|
|
|
err = validateRemoteStateCompatibility(bytes.NewReader(data))
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if !isLocalStateStale(local, bytes.NewReader(data)) {
|
|
|
|
log.Infof(ctx, "Local deployment state is the same or newer, ignoring remote state")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Truncating the file before writing
|
|
|
|
err = local.Truncate(0)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-12-11 12:26:00 +00:00
|
|
|
}
|
2024-03-18 14:41:58 +00:00
|
|
|
_, err = local.Seek(0, 0)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-12-11 12:26:00 +00:00
|
|
|
}
|
2024-03-18 14:41:58 +00:00
|
|
|
|
|
|
|
// Write file to disk.
|
|
|
|
log.Infof(ctx, "Writing remote deployment state file to local cache directory")
|
|
|
|
_, err = io.Copy(local, bytes.NewReader(data))
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var state DeploymentState
|
|
|
|
err = json.Unmarshal(data, &state)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new snapshot based on the deployment state file.
|
2024-04-18 15:13:16 +00:00
|
|
|
opts, err := files.GetSyncOptions(ctx, bundle.ReadOnly(b))
|
2024-03-18 14:41:58 +00:00
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
log.Infof(ctx, "Creating new snapshot")
|
2024-08-21 15:33:25 +00:00
|
|
|
snapshot, err := sync.NewSnapshot(state.Files.ToSlice(b.SyncRoot), opts)
|
2024-03-18 14:41:58 +00:00
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Persist the snapshot to disk.
|
|
|
|
log.Infof(ctx, "Persisting snapshot to disk")
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(snapshot.Save(ctx))
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *statePull) remoteState(ctx context.Context, f filer.Filer) (*bytes.Buffer, error) {
|
|
|
|
// Download deployment state file from filer to local cache directory.
|
|
|
|
remote, err := f.Read(ctx, DeploymentStateFileName)
|
|
|
|
if err != nil {
|
|
|
|
// On first deploy this file doesn't yet exist.
|
|
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer remote.Close()
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
_, err = io.Copy(&buf, remote)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &buf, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *statePull) Name() string {
|
|
|
|
return "deploy:state-pull"
|
|
|
|
}
|
|
|
|
|
|
|
|
// StatePull returns a mutator that pulls the deployment state from the Databricks workspace
|
|
|
|
func StatePull() bundle.Mutator {
|
|
|
|
return &statePull{StateFiler}
|
|
|
|
}
|