package sync import ( "context" "crypto/md5" "encoding/hex" "encoding/json" "errors" "fmt" "io/fs" "os" "path/filepath" "time" "github.com/databricks/cli/libs/fileset" "github.com/databricks/cli/libs/log" ) // Bump it up every time a potentially breaking change is made to the snapshot schema const LatestSnapshotVersion = "v1" // A snapshot is a persistant store of knowledge this CLI has about state of files // in the remote repo. We use the last modified times (mtime) of files to determine // whether a files need to be updated in the remote repo. // // 1. Any stale files in the remote repo are updated. That is if the last modified // time recorded in the snapshot is less than the actual last modified time of the file // // 2. Any files present in snapshot but absent locally are deleted from remote path // // Changing either the databricks workspace (ie Host) or the remote path (ie RemotePath) // local files are being synced to will make this CLI switch to a different // snapshot for persisting/loading sync state type Snapshot struct { // Path where this snapshot was loaded from and will be saved to. // Intentionally not part of the snapshot state because it may be moved by the user. snapshotPath string // New indicates if this is a fresh snapshot or if it was loaded from disk. New bool `json:"-"` // version for snapshot schema. Only snapshots matching the latest snapshot // schema version are used and older ones are invalidated (by deleting them) Version string `json:"version"` // hostname of the workspace this snapshot is for Host string `json:"host"` // Path in workspace for project repo RemotePath string `json:"remote_path"` *SnapshotState } const syncSnapshotDirName = "sync-snapshots" func NewSnapshot(localFiles []fileset.File, opts *SyncOptions) (*Snapshot, error) { snapshotPath, err := SnapshotPath(opts) if err != nil { return nil, err } snapshotState, err := NewSnapshotState(localFiles) if err != nil { return nil, err } // Reset last modified times to make sure all files are synced snapshotState.ResetLastModifiedTimes() return &Snapshot{ snapshotPath: snapshotPath, New: true, Version: LatestSnapshotVersion, Host: opts.Host, RemotePath: opts.RemotePath, SnapshotState: snapshotState, }, nil } func GetFileName(host, remotePath string) string { hash := md5.Sum([]byte(host + remotePath)) hashString := hex.EncodeToString(hash[:]) return hashString[:16] + ".json" } // Compute path of the snapshot file on the local machine // The file name for unique for a tuple of (host, remotePath) // precisely it's the first 16 characters of md5(concat(host, remotePath)) func SnapshotPath(opts *SyncOptions) (string, error) { snapshotDir := filepath.Join(opts.SnapshotBasePath, syncSnapshotDirName) if _, err := os.Stat(snapshotDir); errors.Is(err, fs.ErrNotExist) { err = os.MkdirAll(snapshotDir, 0o755) if err != nil { return "", fmt.Errorf("failed to create config directory: %s", err) } } fileName := GetFileName(opts.Host, opts.RemotePath) return filepath.Join(snapshotDir, fileName), nil } func newSnapshot(ctx context.Context, opts *SyncOptions) (*Snapshot, error) { path, err := SnapshotPath(opts) if err != nil { return nil, err } return &Snapshot{ snapshotPath: path, New: true, Version: LatestSnapshotVersion, Host: opts.Host, RemotePath: opts.RemotePath, SnapshotState: &SnapshotState{ LastModifiedTimes: make(map[string]time.Time), LocalToRemoteNames: make(map[string]string), RemoteToLocalNames: make(map[string]string), }, }, nil } func (s *Snapshot) Save(ctx context.Context) error { f, err := os.OpenFile(s.snapshotPath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0o644) if err != nil { return fmt.Errorf("failed to create/open persisted sync snapshot file: %s", err) } defer f.Close() // persist snapshot to disk bytes, err := json.MarshalIndent(s, "", " ") if err != nil { return fmt.Errorf("failed to json marshal in-memory snapshot: %s", err) } _, err = f.Write(bytes) if err != nil { return fmt.Errorf("failed to write sync snapshot to disk: %s", err) } return nil } func loadOrNewSnapshot(ctx context.Context, opts *SyncOptions) (*Snapshot, error) { snapshot, err := newSnapshot(ctx, opts) if err != nil { return nil, err } // Snapshot file not found. We return the new copy. if _, err := os.Stat(snapshot.snapshotPath); errors.Is(err, fs.ErrNotExist) { return snapshot, nil } bytes, err := os.ReadFile(snapshot.snapshotPath) if err != nil { return nil, fmt.Errorf("failed to read sync snapshot from disk: %s", err) } var fromDisk Snapshot err = json.Unmarshal(bytes, &fromDisk) if err != nil { return nil, fmt.Errorf("failed to json unmarshal persisted snapshot: %s", err) } // invalidate old snapshot with schema versions if fromDisk.Version != LatestSnapshotVersion { log.Warnf(ctx, "Did not load existing snapshot because its version is %s while the latest version is %s", snapshot.Version, LatestSnapshotVersion) return newSnapshot(ctx, opts) } // unmarshal again over the existing snapshot instance err = json.Unmarshal(bytes, &snapshot) if err != nil { return nil, fmt.Errorf("failed to json unmarshal persisted snapshot: %s", err) } // Ensure that all paths are slash-separated upon loading // an existing snapshot file. If it was created by an older // CLI version (<= v0.220.0), it may contain backslashes. snapshot.SnapshotState = snapshot.SnapshotState.ToSlash() snapshot.New = false return snapshot, nil } func (s *Snapshot) diff(ctx context.Context, all []fileset.File) (diff, error) { targetState, err := NewSnapshotState(all) if err != nil { return diff{}, fmt.Errorf("error while computing new sync state: %w", err) } currentState := s.SnapshotState if err := currentState.validate(); err != nil { return diff{}, fmt.Errorf("error parsing existing sync state. Please delete your existing sync snapshot file (%s) and retry: %w", s.snapshotPath, err) } // Compute diff to apply to get from current state to new target state. diff := computeDiff(targetState, currentState) // Update state to new value. This is not persisted to the file system before // the diff is applied successfully. s.SnapshotState = targetState return diff, nil }