2024-03-18 14:41:58 +00:00
|
|
|
package deploy
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
2024-06-03 12:39:36 +00:00
|
|
|
"errors"
|
2024-03-18 14:41:58 +00:00
|
|
|
"io"
|
2024-06-03 12:39:36 +00:00
|
|
|
"io/fs"
|
2024-03-18 14:41:58 +00:00
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/databricks/cli/bundle"
|
|
|
|
"github.com/databricks/cli/internal/build"
|
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/log"
|
2024-07-16 10:01:58 +00:00
|
|
|
"github.com/google/uuid"
|
2024-03-18 14:41:58 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type stateUpdate struct{}
|
|
|
|
|
|
|
|
func (s *stateUpdate) Name() string {
|
|
|
|
return "deploy:state-update"
|
|
|
|
}
|
|
|
|
|
2024-03-25 14:18:47 +00:00
|
|
|
func (s *stateUpdate) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
2024-03-18 14:41:58 +00:00
|
|
|
state, err := load(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
|
|
|
}
|
|
|
|
|
|
|
|
// Increment the state sequence.
|
|
|
|
state.Seq = state.Seq + 1
|
|
|
|
|
|
|
|
// Update timestamp.
|
|
|
|
state.Timestamp = time.Now().UTC()
|
|
|
|
|
|
|
|
// Update the CLI version and deployment state version.
|
|
|
|
state.CliVersion = build.GetInfo().Version
|
|
|
|
state.Version = DeploymentStateVersion
|
|
|
|
|
2024-06-17 09:48:52 +00:00
|
|
|
// Update the state with the current list of synced files.
|
|
|
|
fl, err := FromSlice(b.Files)
|
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
|
|
|
}
|
|
|
|
state.Files = fl
|
|
|
|
|
2024-07-16 10:01:58 +00:00
|
|
|
// Generate a UUID for the deployment, if one does not already exist
|
|
|
|
if state.ID == uuid.Nil {
|
|
|
|
state.ID = uuid.New()
|
|
|
|
}
|
|
|
|
|
2024-03-18 14:41:58 +00:00
|
|
|
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
|
|
|
}
|
|
|
|
// Write the state back to the file.
|
|
|
|
f, err := os.OpenFile(statePath, os.O_CREATE|os.O_RDWR|os.O_TRUNC, 0o600)
|
|
|
|
if err != nil {
|
|
|
|
log.Infof(ctx, "Unable to open 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
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
data, err := json.Marshal(state)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2024-03-18 14:41:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_, err = io.Copy(f, 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
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func StateUpdate() bundle.Mutator {
|
|
|
|
return &stateUpdate{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func load(ctx context.Context, b *bundle.Bundle) (*DeploymentState, error) {
|
|
|
|
// If the file does not exist, return a new DeploymentState.
|
|
|
|
statePath, err := getPathToStateFile(ctx, b)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Infof(ctx, "Loading deployment state from %s", statePath)
|
|
|
|
f, err := os.Open(statePath)
|
|
|
|
if err != nil {
|
2024-06-03 12:39:36 +00:00
|
|
|
if errors.Is(err, fs.ErrNotExist) {
|
2024-03-18 14:41:58 +00:00
|
|
|
log.Infof(ctx, "No deployment state file found")
|
|
|
|
return &DeploymentState{
|
|
|
|
Version: DeploymentStateVersion,
|
|
|
|
CliVersion: build.GetInfo().Version,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
return loadState(f)
|
|
|
|
}
|