zord-tree/tree.go

499 lines
9.7 KiB
Go
Raw Permalink Normal View History

2021-05-04 20:26:41 +00:00
package zord_tree
import (
"bufio"
2023-06-26 21:44:32 +00:00
"crypto/md5"
"errors"
2021-05-11 22:10:18 +00:00
"fmt"
"g.arns.lt/zordsdavini/abcex"
cp "github.com/otiai10/copy"
2023-06-26 21:44:32 +00:00
"golang.org/x/exp/slices"
"io"
2021-05-04 20:26:41 +00:00
"io/fs"
"io/ioutil"
"os"
"path"
2021-05-11 22:10:18 +00:00
"path/filepath"
2021-05-04 20:26:41 +00:00
"strings"
)
type File struct {
2021-05-12 05:17:43 +00:00
Id string
Name string
FullPath string
Category []string
Tags []string
Meta map[string]string
2021-05-04 20:26:41 +00:00
}
type Tree struct {
2021-05-12 05:17:43 +00:00
Path string
Dirs []Tree
Files []File
2021-05-04 20:26:41 +00:00
}
2023-06-26 21:44:32 +00:00
var readableFormats = []string{".md"}
const attachmentDirName = "__a"
func (t Tree) FileById(id string) (File, error) {
for _, f := range t.Files {
if f.Id == id {
return f, nil
}
}
for _, t2 := range t.Dirs {
f, err := t2.FileById(id)
if err == nil {
return f, nil
}
}
return File{}, errors.New("file was not found")
}
func (t Tree) Slice(path string) (Tree, error) {
if t.Path == path {
return t, nil
}
for _, t2 := range t.Dirs {
t3, err := t2.Slice(path)
if err == nil {
return t3, nil
}
}
return Tree{}, errors.New("tree was not found")
}
2022-08-30 16:39:40 +00:00
func (t Tree) Filter(filter map[string][]string) (Tree, bool) {
filtered := Tree{}
filtered.Path = t.Path
found := false
for _, f := range t.Files {
addFile := false
2022-08-30 16:39:40 +00:00
for option, values := range filter {
for _, value := range values {
if option == "tag" {
for _, tag := range f.Tags {
if tag == value {
addFile = true
}
}
2022-08-30 16:39:40 +00:00
continue
}
if strings.Contains(f.Meta[option], value) {
addFile = true
}
}
}
if addFile {
found = true
filtered.Files = append(filtered.Files, f)
}
}
for _, t2 := range t.Dirs {
filteredChild, foundChild := t2.Filter(filter)
if foundChild {
found = true
filtered.Dirs = append(filtered.Dirs, filteredChild)
}
}
return filtered, found
}
2021-05-09 18:17:07 +00:00
func BuildTree(dirPath string, meta []string) (Tree, error) {
return readPath(dirPath, []string{}, meta)
2021-05-04 20:26:41 +00:00
}
func PopulateTree(sourcePath string, destPath string, meta []string, customMeta map[string]func() string) error {
2023-06-26 21:44:32 +00:00
attachmentRegistry, err := moveAttachments(sourcePath)
if err != nil {
return err
}
err = fixFormat(sourcePath, attachmentRegistry)
if err != nil {
return err
}
2022-05-08 20:19:55 +00:00
var id int64 = 0
id, err = getMaxId(sourcePath)
2021-05-11 22:10:18 +00:00
if err != nil {
return err
}
id++
err = addMissingId(sourcePath, id)
if err != nil {
return err
}
err = addMissingMeta(sourcePath, meta, customMeta)
2021-05-11 22:10:18 +00:00
if err != nil {
return err
}
err = cp.Copy(sourcePath, destPath)
return err
}
2023-06-26 21:44:32 +00:00
func moveAttachments(dir string) (map[string]string, error) {
attachmentRegistry := make(map[string]string)
err := filepath.Walk(dir, func(fullPath string, info os.FileInfo, e error) error {
if e != nil {
return e
}
2023-06-26 21:44:32 +00:00
var clearDir = strings.TrimPrefix(dir, "./")
if info.Mode().IsRegular() && !slices.Contains(readableFormats, path.Ext(fullPath)) && !strings.HasPrefix(fullPath, fmt.Sprintf("%s/%s", clearDir, attachmentDirName)) {
f, err := os.Open(fullPath)
if err != nil {
return err
}
defer f.Close()
h := md5.New()
if _, err := io.Copy(h, f); err != nil {
return err
}
_ = os.Mkdir(fmt.Sprintf("%s/%s", dir, attachmentDirName), 0755)
attachmentRegistry[fullPath] = fmt.Sprintf("%s/%x%s", attachmentDirName, h.Sum(nil), path.Ext(fullPath))
newPath := fmt.Sprintf("%s/%s/%x%s", dir, attachmentDirName, h.Sum(nil), path.Ext(fullPath))
err = os.Rename(fullPath, newPath)
if err != nil {
return err
}
}
return nil
})
return attachmentRegistry, err
}
func fixFormat(dir string, attachmentRegistry map[string]string) error {
err := filepath.Walk(dir, func(fullPath string, info os.FileInfo, e error) error {
if e != nil {
return e
}
if info.Mode().IsRegular() && slices.Contains(readableFormats, path.Ext(fullPath)) {
osf, err := os.Open(fullPath)
if err != nil {
return err
}
// remove all empty lines and separate split line
content := ""
format := true
scanner := bufio.NewScanner(osf)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
line := scanner.Text()
if line == "---" {
format = false
}
2022-05-08 20:19:55 +00:00
if format {
line = strings.Trim(line, " ")
2022-05-08 20:19:55 +00:00
if line != "" {
content = content + "\n" + line
}
} else {
content = content + "\n" + line
}
}
2023-06-26 21:44:32 +00:00
err = ioutil.WriteFile(fullPath, []byte(content), 0644)
// format split line
2023-06-26 21:44:32 +00:00
b, err := ioutil.ReadFile(fullPath) // just pass the file name
2022-05-08 21:58:00 +00:00
if err != nil {
return err
}
str := string(b)
str = strings.Replace(str, "\n---\n", "\n\n---\n", 1)
2023-06-26 21:44:32 +00:00
err = ioutil.WriteFile(fullPath, []byte(str), 0644)
}
return nil
})
return err
}
func addMissingMeta(dir string, meta []string, customMeta map[string]func() string) error {
2021-05-11 22:10:18 +00:00
err := filepath.Walk(dir, func(path string, info os.FileInfo, e error) error {
if e != nil {
return e
}
if info.Mode().IsRegular() {
check := map[string]bool{}
for _, option := range meta {
check[option] = false
}
osf, err := os.Open(path)
if err != nil {
return err
}
scanner := bufio.NewScanner(osf)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
line := scanner.Text()
for _, option := range meta {
2022-07-30 06:36:59 +00:00
if strings.HasPrefix(line, "* "+option+":") {
2021-05-11 22:10:18 +00:00
check[option] = true
}
}
if line == "---" {
for option, process := range check {
if !process {
defaultValue := ""
if _, ok := customMeta[option]; ok {
defaultValue = customMeta[option]()
}
err = addMeta(path, option, defaultValue)
2021-05-11 22:10:18 +00:00
if err != nil {
return err
}
}
}
break
}
}
}
return nil
})
2022-05-08 20:19:55 +00:00
return err
2021-05-11 22:10:18 +00:00
}
func addMissingId(dir string, id int64) error {
err := filepath.Walk(dir, func(path string, info os.FileInfo, e error) error {
if e != nil {
return e
}
if info.Mode().IsRegular() {
osf, err := os.Open(path)
if err != nil {
return err
}
scanner := bufio.NewScanner(osf)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
line := scanner.Text()
2022-08-01 14:16:56 +00:00
if strings.HasPrefix(line, "* id:") {
2021-05-11 22:10:18 +00:00
break
}
if line == "---" {
err = addMeta(path, "id", abcex.Encode(id))
if err != nil {
return err
}
id++
break
}
}
}
return nil
})
if err != nil {
return err
}
2022-05-08 20:19:55 +00:00
return nil
2021-05-11 22:10:18 +00:00
}
func addMeta(path string, option string, value string) error {
b, err := ioutil.ReadFile(path) // just pass the file name
if err != nil {
2022-05-08 20:19:55 +00:00
return err
2021-05-11 22:10:18 +00:00
}
str := string(b)
2022-07-30 06:36:59 +00:00
str = strings.Replace(str, "\n\n---\n", fmt.Sprintf("\n* %s: %s\n\n---\n", option, value), 1)
2021-05-11 22:10:18 +00:00
err = ioutil.WriteFile(path, []byte(str), 0644)
2022-05-08 20:19:55 +00:00
return err
2021-05-11 22:10:18 +00:00
}
func getMaxId(dir string) (int64, error) {
var max int64 = 0
err := filepath.Walk(dir, func(path string, info os.FileInfo, e error) error {
if e != nil {
return e
}
if info.Mode().IsRegular() {
osf, err := os.Open(path)
if err != nil {
return err
}
scanner := bufio.NewScanner(osf)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
line := scanner.Text()
if line == "---" {
break
}
2022-08-01 14:16:56 +00:00
if strings.HasPrefix(line, "* id:") {
line = strings.TrimPrefix(line, "* id:")
2021-05-11 22:10:18 +00:00
i := abcex.Decode(strings.Trim(line, " "))
if i > max {
max = i
}
}
}
}
return nil
})
if err != nil {
return max, err
}
return max, nil
}
func readPath(dirPath string, category []string, meta []string) (Tree, error) {
2023-06-26 21:44:32 +00:00
var clearDir = strings.TrimPrefix(dirPath, "./")
2021-05-04 20:26:41 +00:00
2023-06-26 21:44:32 +00:00
tree := Tree{}
tree.Path = clearDir
2021-05-04 20:26:41 +00:00
files, err := ioutil.ReadDir(dirPath)
if err != nil {
return tree, err
}
for _, file := range files {
fullPath := path.Join(dirPath, file.Name())
if file.IsDir() {
2023-06-26 21:44:32 +00:00
if strings.HasPrefix(fullPath, fmt.Sprintf("%s/%s", clearDir, attachmentDirName)) {
continue
}
2021-05-09 18:17:07 +00:00
nextDir, err := readPath(fullPath, append(category, file.Name()), meta)
2021-05-04 20:26:41 +00:00
if err != nil {
return tree, err
}
2021-05-12 05:17:43 +00:00
tree.Dirs = append(tree.Dirs, nextDir)
2021-05-04 20:26:41 +00:00
continue
}
_, err := ioutil.ReadFile(fullPath)
if err != nil {
return tree, err
}
2021-05-09 18:17:07 +00:00
nextFile, err := readFile(file, fullPath, category, meta)
2021-05-04 20:26:41 +00:00
if err != nil {
return tree, err
}
2021-05-12 05:17:43 +00:00
tree.Files = append(tree.Files, nextFile)
2021-05-04 20:26:41 +00:00
}
return tree, nil
}
2021-05-09 18:17:07 +00:00
func readFile(file fs.FileInfo, fullPath string, category []string, meta []string) (File, error) {
2021-05-04 20:26:41 +00:00
f := File{
2021-05-12 05:17:43 +00:00
Name: file.Name(),
FullPath: fullPath,
Category: category,
Meta: map[string]string{},
2021-05-04 20:26:41 +00:00
}
osf, err := os.Open(fullPath)
if err != nil {
return File{}, err
}
scanner := bufio.NewScanner(osf)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
line := scanner.Text()
if line == "---" {
break
}
2022-07-30 06:36:59 +00:00
if strings.HasPrefix(line, "* tags:") {
line = strings.TrimPrefix(line, "* tags:")
2021-05-04 20:26:41 +00:00
t := strings.Split(line, ",")
tags := []string{}
for _, tag := range t {
tags = append(tags, strings.Trim(tag, " "))
}
2021-05-12 05:17:43 +00:00
f.Tags = tags
}
2022-07-30 06:36:59 +00:00
if strings.HasPrefix(line, "* id:") {
line = strings.TrimPrefix(line, "* id:")
2021-05-12 05:17:43 +00:00
f.Id = strings.Trim(line, " ")
2021-05-04 20:26:41 +00:00
}
2021-05-09 18:17:07 +00:00
for _, option := range meta {
2022-07-30 06:36:59 +00:00
if strings.HasPrefix(line, "* "+option) {
line = strings.TrimPrefix(line, "* "+option+":")
2021-05-12 05:17:43 +00:00
f.Meta[option] = strings.Trim(line, " ")
2021-05-09 18:17:07 +00:00
}
2021-05-04 20:26:41 +00:00
}
}
2022-05-08 20:19:55 +00:00
_ = osf.Close()
2021-05-04 20:26:41 +00:00
return f, nil
}
func ReadFileContent(file File) (string, error) {
2022-05-08 20:19:55 +00:00
osf, err := os.Open(file.FullPath)
if err != nil {
2022-05-08 20:19:55 +00:00
return "", err
}
content := ""
2022-05-08 21:58:00 +00:00
separator := ""
removeEmptyLine := true
isMetaPart := true
scanner := bufio.NewScanner(osf)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
line := scanner.Text()
if line == "---" {
isMetaPart = false
continue
}
2022-05-08 20:19:55 +00:00
if isMetaPart {
continue
}
2022-05-08 20:19:55 +00:00
if removeEmptyLine {
line = strings.Trim(line, " ")
2022-05-08 20:19:55 +00:00
if line == "" {
continue
} else {
removeEmptyLine = false
}
}
2022-05-08 21:58:00 +00:00
content = content + separator + line
if separator == "" {
separator = "\n"
}
}
2022-05-08 20:19:55 +00:00
_ = osf.Close()
return content, nil
}