yaml namespaces
parent
0121c426f7
commit
bc8a0d82df
|
|
@ -103,6 +103,12 @@ func TestImplementations(t *testing.T) {
|
||||||
cases = append(cases, bolt)
|
cases = append(cases, bolt)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if yamls, err := NewYaml(path.Join(dir, "yaml")); err != nil {
|
||||||
|
t.Errorf("cannot make yaml: %v", err)
|
||||||
|
} else {
|
||||||
|
cases = append(cases, yamls)
|
||||||
|
}
|
||||||
|
|
||||||
if files, err := NewFiles(path.Join(dir, "files")); err != nil {
|
if files, err := NewFiles(path.Join(dir, "files")); err != nil {
|
||||||
t.Errorf("cannot make files: %v", err)
|
t.Errorf("cannot make files: %v", err)
|
||||||
} else {
|
} else {
|
||||||
|
|
|
||||||
1
go.mod
1
go.mod
|
|
@ -14,6 +14,7 @@ require (
|
||||||
github.com/patrickmn/go-cache v2.1.0+incompatible
|
github.com/patrickmn/go-cache v2.1.0+incompatible
|
||||||
github.com/syndtr/goleveldb v1.0.0
|
github.com/syndtr/goleveldb v1.0.0
|
||||||
go.mongodb.org/mongo-driver v1.7.2
|
go.mongodb.org/mongo-driver v1.7.2
|
||||||
|
gopkg.in/yaml.v2 v2.2.8 // indirect
|
||||||
local/logb v0.0.0-00010101000000-000000000000
|
local/logb v0.0.0-00010101000000-000000000000
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,129 @@
|
||||||
|
package storage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"errors"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
yaml "gopkg.in/yaml.v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
yamlExt = ".yaml"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Yaml struct {
|
||||||
|
path string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewYaml(p string) (*Yaml, error) {
|
||||||
|
_, err := os.Stat(path.Dir(p))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
p, err = filepath.Abs(p)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Yaml{
|
||||||
|
path: p,
|
||||||
|
}, os.MkdirAll(path.Dir(p), os.ModePerm)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) Namespaces() ([][]string, error) {
|
||||||
|
m, err := y.getMap()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return keysDFS(m)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) List(ns []string, limits ...string) ([]string, error) {
|
||||||
|
return nil, errors.New("not impl")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) Get(key string, ns ...string) ([]byte, error) {
|
||||||
|
r, err := y.GetStream(key, ns...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return ioutil.ReadAll(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) GetStream(key string, ns ...string) (io.Reader, error) {
|
||||||
|
return nil, errors.New("not impl")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) Set(key string, value []byte, ns ...string) error {
|
||||||
|
r := bytes.NewReader(value)
|
||||||
|
if value == nil {
|
||||||
|
return y.Del(key, ns...)
|
||||||
|
}
|
||||||
|
return y.SetStream(key, r, ns...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) Del(key string, ns ...string) error {
|
||||||
|
return errors.New("not impl")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) SetStream(key string, r io.Reader, ns ...string) error {
|
||||||
|
return errors.New("not impl")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) Close() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) getMap() (map[string]interface{}, error) {
|
||||||
|
b, err := y.get()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var m map[string]interface{}
|
||||||
|
err = yaml.Unmarshal(b, &m)
|
||||||
|
return m, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) get() ([]byte, error) {
|
||||||
|
b, err := ioutil.ReadFile(y.path)
|
||||||
|
if err == os.ErrNotExist {
|
||||||
|
return []byte{}, nil
|
||||||
|
}
|
||||||
|
return b, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (y *Yaml) set(b []byte) error {
|
||||||
|
return ioutil.WriteFile(y.path, b, os.ModePerm)
|
||||||
|
}
|
||||||
|
|
||||||
|
func keysDFS(m map[string]interface{}) ([][]string, error) {
|
||||||
|
keys, _, err := _keysDFS(m)
|
||||||
|
return keys, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func _keysDFS(m map[string]interface{}) ([][]string, bool, error) {
|
||||||
|
keys := make([][]string, 0)
|
||||||
|
hasNonMaps := false
|
||||||
|
for k, v := range m {
|
||||||
|
if subm, ok := v.(map[string]interface{}); ok {
|
||||||
|
subkeys, hasElements, err := _keysDFS(subm)
|
||||||
|
if err != nil {
|
||||||
|
return nil, false, err
|
||||||
|
}
|
||||||
|
if hasElements {
|
||||||
|
subkeys = append(subkeys, []string{})
|
||||||
|
}
|
||||||
|
for i := range subkeys {
|
||||||
|
subkeys[i] = append([]string{k}, subkeys[i]...)
|
||||||
|
keys = append(keys, subkeys[i])
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
hasNonMaps = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return keys, hasNonMaps, nil
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,51 @@
|
||||||
|
package storage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestKeysDFS(t *testing.T) {
|
||||||
|
cases := map[string]struct {
|
||||||
|
input map[string]interface{}
|
||||||
|
want [][]string
|
||||||
|
}{
|
||||||
|
"empty": {
|
||||||
|
input: map[string]interface{}{},
|
||||||
|
want: [][]string{},
|
||||||
|
},
|
||||||
|
"top level non map keys": {
|
||||||
|
input: map[string]interface{}{"a": "b", "c": "d"},
|
||||||
|
want: [][]string{},
|
||||||
|
},
|
||||||
|
"top level non map keys and map key": {
|
||||||
|
input: map[string]interface{}{"a": "b", "c": "d", "e": map[string]interface{}{"f": "g"}},
|
||||||
|
want: [][]string{[]string{"e"}},
|
||||||
|
},
|
||||||
|
"top level non map keys and map key and nested, ignore empty nested": {
|
||||||
|
input: map[string]interface{}{
|
||||||
|
"a": "b",
|
||||||
|
"c": "d",
|
||||||
|
"e": map[string]interface{}{
|
||||||
|
"f": map[string]interface{}{"g": "h"},
|
||||||
|
},
|
||||||
|
"i": map[string]interface{}{},
|
||||||
|
"j": map[string]interface{}{"k": "l"},
|
||||||
|
},
|
||||||
|
want: [][]string{[]string{"e", "f"}, []string{"j"}},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for name, d := range cases {
|
||||||
|
c := d
|
||||||
|
t.Run(name, func(t *testing.T) {
|
||||||
|
got, err := keysDFS(c.input)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if fmt.Sprintf("%+v", got) != fmt.Sprintf("%+v", c.want) {
|
||||||
|
t.Fatalf("want: %+v\ngot: %+v", c.want, got)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue