tree is now filetree based

master
Bel LaPointe 2022-02-09 08:49:08 -07:00
parent fc263d6c2d
commit 9ad262e607
5 changed files with 122 additions and 51 deletions

View File

@ -0,0 +1,4 @@
title: Untitled (2022-02-09 15:31:57.033525 +0000 UTC)
deleted: false
updated: 2022-02-09T15:31:57.033525Z
pid: ""

View File

@ -0,0 +1,4 @@
title: Untitled (2022-02-09 15:31:36.139835 +0000 UTC)
deleted: false
updated: 2022-02-09T15:31:36.139835Z
pid: ""

View File

@ -212,7 +212,7 @@ func (server *Server) rootHandler(w http.ResponseWriter, r *http.Request) error
}
func (server *Server) tree() *Tree {
return NewTree(path.Join(server.root, "tree.yaml"))
return NewTree(path.Dir(server.diskFilePath("id")))
}
func (server *Server) diskFilePath(id string) string {

View File

@ -4,13 +4,17 @@ import (
"errors"
"io/ioutil"
"os"
"path"
"path/filepath"
"time"
yaml "gopkg.in/yaml.v2"
)
var errChainNotFound = errors.New("pid chain not found")
type Tree struct {
path string
root string
}
type Branch struct {
@ -25,9 +29,9 @@ type Pretty struct {
Title string
}
func NewTree(path string) *Tree {
func NewTree(root string) *Tree {
return &Tree{
path: path,
root: root,
}
}
@ -74,53 +78,108 @@ func (tree *Tree) GetPretty() (map[string]*Pretty, error) {
}
func (tree *Tree) Get() (map[string]Branch, error) {
b, err := ioutil.ReadFile(tree.path)
if os.IsNotExist(err) {
return map[string]Branch{}, nil
}
m := map[string]Branch{}
err := filepath.Walk(tree.root, func(p string, info os.FileInfo, err error) error {
if err != nil {
return nil, err
return err
}
if info.IsDir() {
return nil
}
if info.Name() != "meta.yaml" {
return nil
}
b, err := ioutil.ReadFile(p)
if err != nil {
return err
}
var branch Branch
if err := yaml.Unmarshal(b, &branch); err != nil {
return err
}
id := path.Base(path.Dir(p))
pidFullPath := path.Dir(path.Dir(p))
if pidFullPath != tree.root {
branch.PID = path.Base(pidFullPath)
} else {
branch.PID = ""
}
m[id] = branch
return nil
})
if os.IsNotExist(err) {
return m, nil
}
var m map[string]Branch
err = yaml.Unmarshal(b, &m)
return m, err
}
func (tree *Tree) Set(m map[string]Branch) error {
b, err := yaml.Marshal(m)
if err != nil {
return err
}
return ioutil.WriteFile(tree.path, b, os.ModePerm)
}
func (tree *Tree) Del(id string) error {
func (tree *Tree) fullIdAndMeta(id string) (string, Branch, error) {
m, err := tree.Get()
if err != nil {
return err
return "", Branch{}, err
}
branch, ok := m[id]
if !ok {
return nil
return "", Branch{}, errChainNotFound
}
branch.Updated = time.Now().UTC()
branch.Deleted = true
m[id] = branch
return tree.Set(m)
p := id
for branch.PID != "" {
p = path.Join(branch.PID, p)
branch, ok = m[branch.PID]
if !ok {
return "", Branch{}, errChainNotFound
}
}
return path.Join(tree.root, p), branch, nil
}
func (tree *Tree) Put(id string, branch Branch) error {
branch.Updated = time.Now().UTC()
if branch.Title == "" {
branch.Title = "Untitled (" + time.Now().UTC().String() + ")"
}
m, err := tree.Get()
func (tree *Tree) putMeta(fullId string, branch Branch) error {
b, err := yaml.Marshal(branch)
if err != nil {
return err
}
if _, ok := m[branch.PID]; !ok && branch.PID != "" {
return errors.New("PID does not exist")
}
m[id] = branch
return tree.Set(m)
return ensureAndWrite(path.Join(fullId, "meta.yaml"), b)
}
func (tree *Tree) Del(id string) error {
fullId, meta, err := tree.fullIdAndMeta(id)
if err != nil {
if err == errChainNotFound {
return nil
}
return err
}
meta.Updated = time.Now().UTC()
meta.Deleted = true
return tree.putMeta(fullId, meta)
}
func (tree *Tree) Put(id string, branch Branch) error {
fullId, meta, err := tree.fullIdAndMeta(id)
if err == errChainNotFound {
if branch.PID != "" {
fullId, _, err = tree.fullIdAndMeta(branch.PID)
fullId = path.Join(fullId, id)
} else {
err = nil
fullId = path.Join(tree.root, fullId, id)
}
}
if err != nil {
return err
}
meta.Updated = time.Now().UTC()
if meta.Title == "" {
meta.Title = "Untitled (" + time.Now().UTC().String() + ")"
}
if branch.Title != "" {
meta.Title = branch.Title
}
meta.PID = branch.PID
meta.Deleted = branch.Deleted
return tree.putMeta(fullId, meta)
}

View File

@ -3,7 +3,6 @@ package main
import (
"bytes"
"encoding/json"
"path"
"testing"
"time"
@ -11,41 +10,46 @@ import (
)
func TestTree(t *testing.T) {
path := path.Join(t.TempDir(), "index.yaml")
tree := NewTree(path)
tree := NewTree(t.TempDir())
t.Logf("tree.Get() from zero")
if m, err := tree.Get(); err != nil {
t.Fatal(err)
t.Fatal("failed to get empty tree:", err)
} else if m == nil {
t.Fatal(m)
t.Fatal("got a nil tree:", m)
}
t.Logf("tree.Del() from zero")
if err := tree.Del("id"); err != nil {
t.Fatal(err)
t.Fatal("failed to del a nil id:", err)
}
t.Logf("tree.Put(bad pid) from zero")
if err := tree.Put("id", Branch{PID: "fake"}); err == nil {
t.Fatal(err)
t.Fatal("failed to put with a fake pid:", err)
}
t.Logf("tree.Put() from zero")
if err := tree.Put("id", Branch{}); err != nil {
t.Fatal(err)
t.Fatal("failed to put with no pid:", err)
} else if branches, err := tree.Get(); err != nil {
t.Fatal(err)
t.Fatal("failed to get after put:", err)
} else if branch, ok := branches["id"]; !ok {
t.Fatal(err)
t.Fatal("got tree without put id:", branches)
} else if branch.Title == "" {
t.Fatal(branch)
t.Fatal("got no default title", branch)
} else if time.Since(branch.Updated) > time.Hour {
t.Fatal(branch)
t.Fatal("got not updated", branch)
} else if branch.Deleted {
t.Fatal(branch)
t.Fatal("got deleted after put", branch)
}
t.Logf("tree.Put(good pid)")
if err := tree.Put("id2", Branch{PID: "id"}); err != nil {
t.Fatal(err)
}
t.Logf("tree.Del(good pid)")
if err := tree.Del("id"); err != nil {
t.Fatal(err)
} else if branches, err := tree.Get(); err != nil {
@ -58,7 +62,7 @@ func TestTree(t *testing.T) {
}
func TestTreePretty(t *testing.T) {
tree := NewTree(path.Join(t.TempDir(), "tree_pretty.yaml"))
tree := NewTree(t.TempDir())
tree.Put("A", Branch{Title: "A", PID: ""})
tree.Put("AA", Branch{Title: "AA", PID: "A", Deleted: true})
tree.Put("B", Branch{Title: "B", PID: ""})