mirror of
https://github.com/zrepl/zrepl.git
synced 2024-11-22 16:34:32 +01:00
115 lines
2.5 KiB
Go
115 lines
2.5 KiB
Go
|
package zfs
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
type VersionType string
|
||
|
|
||
|
const (
|
||
|
Bookmark VersionType = "bookmark"
|
||
|
Snapshot = "snapshot"
|
||
|
)
|
||
|
|
||
|
func (t VersionType) DelimiterChar() string {
|
||
|
switch t {
|
||
|
case Bookmark:
|
||
|
return "#"
|
||
|
case Snapshot:
|
||
|
return "@"
|
||
|
default:
|
||
|
panic(fmt.Sprintf("unexpected VersionType %#v", t))
|
||
|
}
|
||
|
return ""
|
||
|
}
|
||
|
|
||
|
type FilesystemVersion struct {
|
||
|
Type VersionType
|
||
|
|
||
|
// Display name. Should not be used for identification, only for user output
|
||
|
Name string
|
||
|
|
||
|
// GUID as exported by ZFS. Uniquely identifies a snapshot across pools
|
||
|
Guid uint64
|
||
|
|
||
|
// The TXG in which the snapshot was created. For bookmarks,
|
||
|
// this is the GUID of the snapshot it was initially tied to.
|
||
|
CreateTXG uint64
|
||
|
}
|
||
|
|
||
|
func (v FilesystemVersion) ToAbsPath(p DatasetPath) string {
|
||
|
var b bytes.Buffer
|
||
|
b.WriteString(p.ToString())
|
||
|
b.WriteString(v.Type.DelimiterChar())
|
||
|
b.WriteString(v.Name)
|
||
|
return b.String()
|
||
|
}
|
||
|
|
||
|
type FilesystemVersionFilter interface {
|
||
|
Filter(fsv FilesystemVersion) (accept bool, err error)
|
||
|
}
|
||
|
|
||
|
func ZFSListFilesystemVersions(fs DatasetPath, filter FilesystemVersionFilter) (res []FilesystemVersion, err error) {
|
||
|
var fieldLines [][]string
|
||
|
fieldLines, err = ZFSList(
|
||
|
[]string{"name", "guid", "createtxg"},
|
||
|
"-r", "-d", "1",
|
||
|
"-t", "bookmark,snapshot",
|
||
|
"-s", "createtxg", fs.ToString())
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
res = make([]FilesystemVersion, 0, len(fieldLines))
|
||
|
for _, line := range fieldLines {
|
||
|
|
||
|
if len(line[0]) < 3 {
|
||
|
err = errors.New(fmt.Sprintf("snapshot or bookmark name implausibly short: %s", line[0]))
|
||
|
return
|
||
|
}
|
||
|
|
||
|
snapSplit := strings.SplitN(line[0], "@", 2)
|
||
|
bookmarkSplit := strings.SplitN(line[0], "#", 2)
|
||
|
if len(snapSplit)*len(bookmarkSplit) != 2 {
|
||
|
err = errors.New(fmt.Sprintf("dataset cannot be snapshot and bookmark at the same time: %s", line[0]))
|
||
|
return
|
||
|
}
|
||
|
|
||
|
var v FilesystemVersion
|
||
|
if len(snapSplit) == 2 {
|
||
|
v.Name = snapSplit[1]
|
||
|
v.Type = Snapshot
|
||
|
} else {
|
||
|
v.Name = bookmarkSplit[1]
|
||
|
v.Type = Bookmark
|
||
|
}
|
||
|
|
||
|
if v.Guid, err = strconv.ParseUint(line[1], 10, 64); err != nil {
|
||
|
err = errors.New(fmt.Sprintf("cannot parse GUID: %s", err.Error()))
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if v.CreateTXG, err = strconv.ParseUint(line[2], 10, 64); err != nil {
|
||
|
err = errors.New(fmt.Sprintf("cannot parse CreateTXG: %s", err.Error()))
|
||
|
return
|
||
|
}
|
||
|
|
||
|
accept := true
|
||
|
if filter != nil {
|
||
|
accept, err = filter.Filter(v)
|
||
|
if err != nil {
|
||
|
err = fmt.Errorf("error executing filter: %s", err)
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
if accept {
|
||
|
res = append(res, v)
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return
|
||
|
}
|