mirror of
https://github.com/zrepl/zrepl.git
synced 2025-01-12 09:18:26 +01:00
9465b593f9
We lost the nice context-stack [jobname][taskname][...] at the beginning of each log line when switching to logrus. Define some field names that define these contexts. Write a human-friendly formatter that presents these field names like the solution we had before logrus. Write some other formatters for logfmt and json output along the way. Limit ourselves to stdout logging for now.
104 lines
2.5 KiB
Go
104 lines
2.5 KiB
Go
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"github.com/zrepl/zrepl/zfs"
|
|
"time"
|
|
)
|
|
|
|
type Pruner struct {
|
|
Now time.Time
|
|
DryRun bool
|
|
DatasetFilter zfs.DatasetFilter
|
|
SnapshotPrefix string
|
|
PrunePolicy PrunePolicy
|
|
}
|
|
|
|
type PruneResult struct {
|
|
Filesystem *zfs.DatasetPath
|
|
All []zfs.FilesystemVersion
|
|
Keep []zfs.FilesystemVersion
|
|
Remove []zfs.FilesystemVersion
|
|
}
|
|
|
|
func (p *Pruner) Run(ctx context.Context) (r []PruneResult, err error) {
|
|
|
|
log := ctx.Value(contextKeyLog).(Logger)
|
|
|
|
if p.DryRun {
|
|
log.Info("doing dry run")
|
|
}
|
|
|
|
filesystems, err := zfs.ZFSListMapping(p.DatasetFilter)
|
|
if err != nil {
|
|
log.WithError(err).Error("error applying filesystem filter")
|
|
return nil, err
|
|
}
|
|
if len(filesystems) <= 0 {
|
|
log.Info("no filesystems matching filter")
|
|
return nil, err
|
|
}
|
|
|
|
r = make([]PruneResult, 0, len(filesystems))
|
|
|
|
for _, fs := range filesystems {
|
|
|
|
log := log.WithField(logFSField, fs.ToString())
|
|
|
|
fsversions, err := zfs.ZFSListFilesystemVersions(fs, &PrefixSnapshotFilter{p.SnapshotPrefix})
|
|
if err != nil {
|
|
log.WithError(err).Error("error listing filesytem versions")
|
|
continue
|
|
}
|
|
if len(fsversions) == 0 {
|
|
log.WithField("prefix", p.SnapshotPrefix).Info("no filesystem versions matching prefix")
|
|
continue
|
|
}
|
|
|
|
keep, remove, err := p.PrunePolicy.Prune(fs, fsversions)
|
|
if err != nil {
|
|
log.WithError(err).Error("error evaluating prune policy")
|
|
continue
|
|
}
|
|
|
|
log.WithField("fsversions", fsversions).
|
|
WithField("keep", keep).
|
|
WithField("remove", remove).
|
|
Debug("prune policy debug dump")
|
|
|
|
r = append(r, PruneResult{fs, fsversions, keep, remove})
|
|
|
|
makeFields := func(v zfs.FilesystemVersion) (fields map[string]interface{}) {
|
|
fields = make(map[string]interface{})
|
|
fields["version"] = v.ToAbsPath(fs)
|
|
timeSince := v.Creation.Sub(p.Now)
|
|
fields["age_ns"] = timeSince
|
|
const day time.Duration = 24 * time.Hour
|
|
days := timeSince / day
|
|
remainder := timeSince % day
|
|
fields["age_str"] = fmt.Sprintf("%dd%s", days, remainder)
|
|
return
|
|
}
|
|
|
|
for _, v := range remove {
|
|
fields := makeFields(v)
|
|
log.WithFields(fields).Info("destroying version")
|
|
// echo what we'll do and exec zfs destroy if not dry run
|
|
// TODO special handling for EBUSY (zfs hold)
|
|
// TODO error handling for clones? just echo to cli, skip over, and exit with non-zero status code (we're idempotent)
|
|
if !p.DryRun {
|
|
err := zfs.ZFSDestroyFilesystemVersion(fs, v)
|
|
if err != nil {
|
|
// handle
|
|
log.WithFields(fields).WithError(err).Error("error destroying version")
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|