2017-04-26 17:39:16 +02:00
|
|
|
package zfs
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
2017-04-26 20:25:53 +02:00
|
|
|
"errors"
|
2017-04-26 17:39:16 +02:00
|
|
|
"fmt"
|
2017-04-26 20:25:53 +02:00
|
|
|
"io"
|
|
|
|
"os/exec"
|
2017-04-26 17:39:16 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type DatasetMapping interface {
|
|
|
|
Map(source DatasetPath) (target DatasetPath, err error)
|
|
|
|
}
|
|
|
|
|
2017-05-01 20:35:04 +02:00
|
|
|
func ZFSListMapping(mapping DatasetMapping) (datasets []DatasetPath, err error) {
|
|
|
|
|
|
|
|
if mapping == nil {
|
|
|
|
panic("mapping must not be nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
var lines [][]string
|
|
|
|
lines, err = ZFSList([]string{"name"}, "-r", "-t", "filesystem,volume")
|
|
|
|
|
|
|
|
datasets = make([]DatasetPath, len(lines))
|
|
|
|
|
|
|
|
for i, line := range lines {
|
|
|
|
|
|
|
|
var path DatasetPath
|
|
|
|
if path, err = NewDatasetPath(line[0]); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
_, mapErr := mapping.Map(path)
|
|
|
|
if mapErr != nil && err != NoMatchError {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if mapErr == nil {
|
|
|
|
datasets[i] = path
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-04-26 17:39:16 +02:00
|
|
|
type GlobMapping struct {
|
|
|
|
PrefixPath DatasetPath
|
|
|
|
TargetRoot DatasetPath
|
|
|
|
}
|
|
|
|
|
|
|
|
var NoMatchError error = errors.New("no match found in mapping")
|
|
|
|
|
|
|
|
func (m GlobMapping) Map(source DatasetPath) (target DatasetPath, err error) {
|
|
|
|
|
|
|
|
if len(source) < len(m.PrefixPath) {
|
|
|
|
err = NoMatchError
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-04-26 20:25:53 +02:00
|
|
|
target = make([]string, 0, len(source)+len(m.TargetRoot))
|
2017-04-26 17:39:16 +02:00
|
|
|
target = append(target, m.TargetRoot...)
|
|
|
|
|
|
|
|
for si, sc := range source {
|
|
|
|
target = append(target, sc)
|
|
|
|
if si < len(m.PrefixPath) {
|
|
|
|
if sc != m.PrefixPath[si] {
|
|
|
|
err = NoMatchError
|
|
|
|
return
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
type ComboMapping struct {
|
|
|
|
Mappings []DatasetMapping
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m ComboMapping) Map(source DatasetPath) (target DatasetPath, err error) {
|
|
|
|
for _, sm := range m.Mappings {
|
|
|
|
target, err = sm.Map(source)
|
|
|
|
if err == nil {
|
|
|
|
return target, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, NoMatchError
|
|
|
|
}
|
|
|
|
|
|
|
|
type DirectMapping struct {
|
|
|
|
Source DatasetPath
|
|
|
|
Target DatasetPath
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m DirectMapping) Map(source DatasetPath) (target DatasetPath, err error) {
|
2017-04-26 18:36:01 +02:00
|
|
|
|
|
|
|
if m.Source == nil {
|
|
|
|
return m.Target, nil
|
|
|
|
}
|
|
|
|
|
2017-04-26 17:39:16 +02:00
|
|
|
if len(m.Source) != len(source) {
|
|
|
|
return nil, NoMatchError
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, c := range source {
|
|
|
|
if c != m.Source[i] {
|
|
|
|
return nil, NoMatchError
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.Target, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type ExecMapping struct {
|
|
|
|
Name string
|
|
|
|
Args []string
|
|
|
|
}
|
|
|
|
|
2017-04-26 20:25:53 +02:00
|
|
|
func NewExecMapping(name string, args ...string) (m *ExecMapping) {
|
2017-04-26 17:39:16 +02:00
|
|
|
m = &ExecMapping{
|
|
|
|
Name: name,
|
|
|
|
Args: args,
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m ExecMapping) Map(source DatasetPath) (target DatasetPath, err error) {
|
|
|
|
|
|
|
|
var stdin io.Writer
|
|
|
|
var stdout io.Reader
|
|
|
|
|
|
|
|
cmd := exec.Command(m.Name, m.Args...)
|
|
|
|
|
|
|
|
if stdin, err = cmd.StdinPipe(); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if stdout, err = cmd.StdoutPipe(); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
resp := bufio.NewScanner(stdout)
|
|
|
|
|
|
|
|
if err = cmd.Start(); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
err := cmd.Wait()
|
|
|
|
if err != nil {
|
2017-05-01 20:35:04 +02:00
|
|
|
panic(err)
|
|
|
|
// fmt.Printf("error: %v\n", err) // TODO
|
2017-04-26 17:39:16 +02:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2017-04-26 20:25:53 +02:00
|
|
|
if _, err = io.WriteString(stdin, source.ToString()+"\n"); err != nil {
|
2017-04-26 17:39:16 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if !resp.Scan() {
|
|
|
|
err = errors.New(fmt.Sprintf("unexpected end of file: %v", resp.Err()))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
t := resp.Text()
|
|
|
|
|
|
|
|
switch {
|
2017-04-26 20:25:53 +02:00
|
|
|
case t == "NOMAP":
|
|
|
|
return nil, NoMatchError
|
2017-04-26 17:39:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
target = toDatasetPath(t) // TODO discover garbage?
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|