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 { 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 { func (server *Server) diskFilePath(id string) string {

View File

@ -4,13 +4,17 @@ import (
"errors" "errors"
"io/ioutil" "io/ioutil"
"os" "os"
"path"
"path/filepath"
"time" "time"
yaml "gopkg.in/yaml.v2" yaml "gopkg.in/yaml.v2"
) )
var errChainNotFound = errors.New("pid chain not found")
type Tree struct { type Tree struct {
path string root string
} }
type Branch struct { type Branch struct {
@ -25,9 +29,9 @@ type Pretty struct {
Title string Title string
} }
func NewTree(path string) *Tree { func NewTree(root string) *Tree {
return &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) { func (tree *Tree) Get() (map[string]Branch, error) {
b, err := ioutil.ReadFile(tree.path) m := map[string]Branch{}
if os.IsNotExist(err) { err := filepath.Walk(tree.root, func(p string, info os.FileInfo, err error) error {
return map[string]Branch{}, nil
}
if err != nil { 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 return m, err
} }
func (tree *Tree) Set(m map[string]Branch) error { func (tree *Tree) fullIdAndMeta(id string) (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 {
m, err := tree.Get() m, err := tree.Get()
if err != nil { if err != nil {
return err return "", Branch{}, err
} }
branch, ok := m[id] branch, ok := m[id]
if !ok { if !ok {
return nil return "", Branch{}, errChainNotFound
} }
branch.Updated = time.Now().UTC() p := id
branch.Deleted = true for branch.PID != "" {
m[id] = branch p = path.Join(branch.PID, p)
return tree.Set(m) 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 { func (tree *Tree) putMeta(fullId string, branch Branch) error {
branch.Updated = time.Now().UTC() b, err := yaml.Marshal(branch)
if branch.Title == "" {
branch.Title = "Untitled (" + time.Now().UTC().String() + ")"
}
m, err := tree.Get()
if err != nil { if err != nil {
return err return err
} }
if _, ok := m[branch.PID]; !ok && branch.PID != "" { return ensureAndWrite(path.Join(fullId, "meta.yaml"), b)
return errors.New("PID does not exist")
} }
m[id] = branch
return tree.Set(m) 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 ( import (
"bytes" "bytes"
"encoding/json" "encoding/json"
"path"
"testing" "testing"
"time" "time"
@ -11,41 +10,46 @@ import (
) )
func TestTree(t *testing.T) { func TestTree(t *testing.T) {
path := path.Join(t.TempDir(), "index.yaml") tree := NewTree(t.TempDir())
tree := NewTree(path)
t.Logf("tree.Get() from zero")
if m, err := tree.Get(); err != nil { if m, err := tree.Get(); err != nil {
t.Fatal(err) t.Fatal("failed to get empty tree:", err)
} else if m == nil { } 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 { 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 { 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 { 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 { } 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 { } else if branch, ok := branches["id"]; !ok {
t.Fatal(err) t.Fatal("got tree without put id:", branches)
} else if branch.Title == "" { } else if branch.Title == "" {
t.Fatal(branch) t.Fatal("got no default title", branch)
} else if time.Since(branch.Updated) > time.Hour { } else if time.Since(branch.Updated) > time.Hour {
t.Fatal(branch) t.Fatal("got not updated", branch)
} else if branch.Deleted { } 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 { if err := tree.Put("id2", Branch{PID: "id"}); err != nil {
t.Fatal(err) t.Fatal(err)
} }
t.Logf("tree.Del(good pid)")
if err := tree.Del("id"); err != nil { if err := tree.Del("id"); err != nil {
t.Fatal(err) t.Fatal(err)
} else if branches, err := tree.Get(); err != nil { } else if branches, err := tree.Get(); err != nil {
@ -58,7 +62,7 @@ func TestTree(t *testing.T) {
} }
func TestTreePretty(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("A", Branch{Title: "A", PID: ""})
tree.Put("AA", Branch{Title: "AA", PID: "A", Deleted: true}) tree.Put("AA", Branch{Title: "AA", PID: "A", Deleted: true})
tree.Put("B", Branch{Title: "B", PID: ""}) tree.Put("B", Branch{Title: "B", PID: ""})