mirror of
https://github.com/osmarks/mycorrhiza.git
synced 2025-01-18 22:52:50 +00:00
Merge pull request #3 from bouncepaw/new-hypha-format-better
New hypha format
This commit is contained in:
commit
01a90add0c
@ -9,8 +9,8 @@ type Genealogy struct {
|
||||
|
||||
func setRelations(hyphae map[string]*Hypha) {
|
||||
for name, h := range hyphae {
|
||||
if _, ok := hyphae[h.ParentName]; ok && h.ParentName != "." {
|
||||
hyphae[h.ParentName].ChildrenNames = append(hyphae[h.ParentName].ChildrenNames, name)
|
||||
if _, ok := hyphae[h.ParentName()]; ok && h.ParentName() != "." {
|
||||
hyphae[h.ParentName()].ChildrenNames = append(hyphae[h.ParentName()].ChildrenNames, name)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
84
hypha.go
84
hypha.go
@ -1,70 +1,52 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"log"
|
||||
"net/http"
|
||||
"strconv"
|
||||
)
|
||||
|
||||
type Hypha struct {
|
||||
// Hypha is physically located here. Most fields below are stored in <path>/mm.ini (mm for metametadata). Its revisions are physically located in <path>/<n>/ subfolders. <n> ∈ [0;∞] with 0 being latest revision, 1 the first.
|
||||
Path string
|
||||
// Every hypha was created at some point
|
||||
CreationTime int `json:"creationTime"`
|
||||
// Hypha has name but it can be changed
|
||||
Name string `json:"name"`
|
||||
// Hypha can be deleted. If it is deleted, it is not indexed by most of the software but still can be recovered at some point.
|
||||
Deleted bool `json:"deleted"`
|
||||
// Fields below are not part of m.ini and are created when traversing the file tree.
|
||||
// Hypha can be a child of any other hypha except its children. The parent hypha is stored in <path>/..
|
||||
ParentName string
|
||||
// Hypha can have any number of children which are stored as subfolders in <path>.
|
||||
FullName string
|
||||
Path string
|
||||
ViewCount int `json:"views"`
|
||||
Deleted bool `json:"deleted"`
|
||||
Revisions map[string]*Revision `json:"revisions"`
|
||||
ChildrenNames []string
|
||||
Revisions []Revision
|
||||
parentName string
|
||||
}
|
||||
|
||||
func (h Hypha) String() string {
|
||||
var revbuf string
|
||||
for _, r := range h.Revisions {
|
||||
revbuf += r.String() + "\n"
|
||||
func (h *Hypha) AddChild(childName string) {
|
||||
h.ChildrenNames = append(h.ChildrenNames, childName)
|
||||
}
|
||||
|
||||
// Used with action=zen|view
|
||||
func (h *Hypha) AsHtml(hyphae map[string]*Hypha, rev string) (string, error) {
|
||||
if "0" == rev {
|
||||
rev = h.NewestRevision()
|
||||
}
|
||||
return fmt.Sprintf("Hypha %v {\n\t"+
|
||||
"path %v\n\t"+
|
||||
"created at %v\n\t"+
|
||||
"child of %v\n\t"+
|
||||
"parent of %v\n\t"+
|
||||
"Having these revisions:\n%v"+
|
||||
"}\n", h.Name, h.Path, h.CreationTime, h.ParentName, h.ChildrenNames,
|
||||
revbuf)
|
||||
r, ok := h.Revisions[rev]
|
||||
if !ok {
|
||||
return "", fmt.Errorf("Hypha %v has no such revision: %v", h.FullName, rev)
|
||||
}
|
||||
html, err := r.AsHtml(hyphae)
|
||||
return html, err
|
||||
}
|
||||
|
||||
func GetRevision(hyphae map[string]*Hypha, hyphaName string, rev string, w http.ResponseWriter) (Revision, bool) {
|
||||
for name, _ := range hyphae {
|
||||
if name == hyphaName {
|
||||
for _, r := range hyphae[name].Revisions {
|
||||
id, err := strconv.Atoi(rev)
|
||||
if err != nil {
|
||||
log.Println("No such revision", rev, "at hypha", hyphaName)
|
||||
w.WriteHeader(http.StatusNotFound)
|
||||
return Revision{}, false
|
||||
}
|
||||
if r.Id == id {
|
||||
return r, true
|
||||
}
|
||||
}
|
||||
func (h *Hypha) Name() string {
|
||||
return h.FullName
|
||||
}
|
||||
|
||||
func (h *Hypha) NewestRevision() string {
|
||||
var largest int
|
||||
for k, _ := range h.Revisions {
|
||||
rev, _ := strconv.Atoi(k)
|
||||
if rev > largest {
|
||||
largest = rev
|
||||
}
|
||||
}
|
||||
return Revision{}, false
|
||||
return strconv.Itoa(largest)
|
||||
}
|
||||
|
||||
// `rev` is the id of revision to render. If it = 0, the last one is rendered. If the revision is not found, an error is returned.
|
||||
func (h Hypha) Render(hyphae map[string]*Hypha, rev int) (ret string, err error) {
|
||||
for _, r := range h.Revisions {
|
||||
if r.Id == rev {
|
||||
return r.Render(hyphae)
|
||||
}
|
||||
}
|
||||
return "", errors.New("Revision was not found")
|
||||
func (h *Hypha) ParentName() string {
|
||||
return h.parentName
|
||||
}
|
||||
|
38
main.go
38
main.go
@ -12,6 +12,23 @@ import (
|
||||
"time"
|
||||
)
|
||||
|
||||
func GetRevision(hyphae map[string]*Hypha, hyphaName string, rev string, w http.ResponseWriter) (Revision, bool) {
|
||||
log.Println("Getting hypha", hyphaName, rev)
|
||||
for name, hypha := range hyphae {
|
||||
if name == hyphaName {
|
||||
if rev == "0" {
|
||||
rev = hypha.NewestRevision()
|
||||
}
|
||||
for id, r := range hypha.Revisions {
|
||||
if rev == id {
|
||||
return *r, true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return Revision{}, false
|
||||
}
|
||||
|
||||
func RevInMap(m map[string]string) string {
|
||||
if val, ok := m["rev"]; ok {
|
||||
return val
|
||||
@ -33,7 +50,7 @@ func HandlerGetBinary(w http.ResponseWriter, r *http.Request) {
|
||||
w.WriteHeader(http.StatusNotFound)
|
||||
return
|
||||
}
|
||||
w.Header().Set("Content-Type", rev.MimeType)
|
||||
w.Header().Set("Content-Type", rev.BinaryMime)
|
||||
w.WriteHeader(http.StatusOK)
|
||||
w.Write(fileContents)
|
||||
log.Println("Showing image of", rev.FullName, rev.Id)
|
||||
@ -52,7 +69,7 @@ func HandlerRaw(w http.ResponseWriter, r *http.Request) {
|
||||
w.WriteHeader(http.StatusNotFound)
|
||||
return
|
||||
}
|
||||
w.Header().Set("Content-Type", "text/plain")
|
||||
w.Header().Set("Content-Type", rev.TextMime)
|
||||
w.WriteHeader(http.StatusOK)
|
||||
w.Write(fileContents)
|
||||
log.Println("Serving text data of", rev.FullName, rev.Id)
|
||||
@ -65,7 +82,7 @@ func HandlerZen(w http.ResponseWriter, r *http.Request) {
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
html, err := rev.Render(hyphae)
|
||||
html, err := rev.AsHtml(hyphae)
|
||||
if err != nil {
|
||||
log.Println("Failed to render", rev.FullName)
|
||||
w.WriteHeader(http.StatusInternalServerError)
|
||||
@ -83,7 +100,7 @@ func HandlerView(w http.ResponseWriter, r *http.Request) {
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
html, err := rev.Render(hyphae)
|
||||
html, err := rev.AsHtml(hyphae)
|
||||
if err != nil {
|
||||
log.Println("Failed to render", rev.FullName)
|
||||
w.WriteHeader(http.StatusInternalServerError)
|
||||
@ -131,7 +148,7 @@ var hyphae map[string]*Hypha
|
||||
func hyphaeAsMap(hyphae []*Hypha) map[string]*Hypha {
|
||||
mh := make(map[string]*Hypha)
|
||||
for _, h := range hyphae {
|
||||
mh[h.Name] = h
|
||||
mh[h.Name()] = h
|
||||
}
|
||||
return mh
|
||||
}
|
||||
@ -147,8 +164,11 @@ func main() {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
hyphae = hyphaeAsMap(recurFindHyphae(rootWikiDir))
|
||||
setRelations(hyphae)
|
||||
log.Println("Welcome to MycorrhizaWiki α")
|
||||
log.Println("Indexing hyphae...")
|
||||
hyphae = recurFindHyphae(rootWikiDir)
|
||||
log.Println("Indexed", len(hyphae), "hyphae. Ready to accept requests.")
|
||||
// setRelations(hyphae)
|
||||
|
||||
// Start server code
|
||||
r := mux.NewRouter()
|
||||
@ -197,8 +217,8 @@ func main() {
|
||||
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
||||
w.WriteHeader(http.StatusOK)
|
||||
for _, v := range hyphae {
|
||||
log.Println("Rendering latest revision of hypha", v.Name)
|
||||
html, err := v.Render(hyphae, 0)
|
||||
log.Println("Rendering latest revision of hypha", v.Name())
|
||||
html, err := v.AsHtml(hyphae, "0")
|
||||
if err != nil {
|
||||
fmt.Fprintln(w, err)
|
||||
}
|
||||
|
129
revision.go
129
revision.go
@ -1,59 +1,44 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"github.com/gomarkdown/markdown"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"net/http"
|
||||
)
|
||||
|
||||
type Revision struct {
|
||||
// Revision is hypha's state at some point in time. Future revisions are not really supported. Most data here is stored in m.ini.
|
||||
Id int
|
||||
// Name used at this revision
|
||||
Name string `json:"name"`
|
||||
// Name of hypha
|
||||
FullName string
|
||||
// Present in every hypha. Stored in t.txt.
|
||||
TextPath string
|
||||
// In at least one markup. Supported ones are "myco", "html", "md", "plain"
|
||||
Markup string `json:"markup"`
|
||||
// Some hyphæ have binary contents such as images. Their presence change hypha's behavior in a lot of ways (see methods' implementations). If stored, it is stored in b (filename "b")
|
||||
BinaryPath string
|
||||
BinaryRequest string
|
||||
// To tell what is meaning of binary content, mimeType for them is stored. If the hypha has no binary content, this field must be "application/x-hypha"
|
||||
MimeType string `json:"mimeType"`
|
||||
// Every revision was created at some point. This field stores the creation time of the latest revision
|
||||
RevisionTime int `json:"createdAt"`
|
||||
// Every hypha has any number of tags
|
||||
Tags []string `json:"tags"`
|
||||
// Current revision is authored by someone
|
||||
RevisionAuthor string `json:"author"`
|
||||
// and has a comment in plain text
|
||||
RevisionComment string `json:"comment"`
|
||||
Id int
|
||||
FullName string
|
||||
Tags []string `json:"tags"`
|
||||
ShortName string `json:"name"`
|
||||
Comment string `json:"comment"`
|
||||
Author string `json:"author"`
|
||||
Time int `json:"time"`
|
||||
TextMime string `json:"text_mime"`
|
||||
BinaryMime string `json:"binary_mime"`
|
||||
TextPath string
|
||||
BinaryPath string
|
||||
}
|
||||
|
||||
func (h Revision) String() string {
|
||||
return fmt.Sprintf(`Revision %v created at %v {
|
||||
name: %v
|
||||
textPath: %v
|
||||
markup: %v
|
||||
binaryPath: %v
|
||||
mimeType: %v
|
||||
tags: %v
|
||||
revisionAuthor: %v
|
||||
revisionComment: %v
|
||||
}`, h.Id, h.RevisionTime, h.Name, h.TextPath, h.Markup, h.BinaryPath, h.MimeType, h.Tags, h.RevisionAuthor, h.RevisionComment)
|
||||
// During initialisation, it is guaranteed that r.BinaryMime is set to "" if the revision has no binary data.
|
||||
func (r *Revision) hasBinaryData() bool {
|
||||
return r.BinaryMime != ""
|
||||
}
|
||||
|
||||
// This method is meant to be called only by Hypha#Render.
|
||||
func (r Revision) Render(hyphae map[string]*Hypha) (ret string, err error) {
|
||||
func (r *Revision) urlOfBinary() string {
|
||||
return fmt.Sprintf("/%s?action=getBinary&rev=%d", r.FullName, r.Id)
|
||||
}
|
||||
|
||||
// TODO: use templates https://github.com/bouncepaw/mycorrhiza/issues/2
|
||||
func (r *Revision) AsHtml(hyphae map[string]*Hypha) (ret string, err error) {
|
||||
ret += `<article class="page">
|
||||
<h1 class="page__title">` + r.FullName + `</h1>
|
||||
`
|
||||
// If it is a binary hypha (we support only images for now):
|
||||
// TODO: support things other than images.
|
||||
if r.MimeType != "application/x-hypha" {
|
||||
ret += fmt.Sprintf(`<img src="/%s" class="page__image"/>`, r.BinaryRequest)
|
||||
// TODO: support things other than images
|
||||
if r.hasBinaryData() {
|
||||
ret += fmt.Sprintf(`<img src="%s" class="page__amnt"/>`, r.urlOfBinary())
|
||||
}
|
||||
|
||||
contents, err := ioutil.ReadFile(r.TextPath)
|
||||
@ -63,17 +48,69 @@ func (r Revision) Render(hyphae map[string]*Hypha) (ret string, err error) {
|
||||
|
||||
// TODO: support more markups.
|
||||
// TODO: support mycorrhiza extensions like transclusion.
|
||||
switch r.Markup {
|
||||
case "plain":
|
||||
ret += fmt.Sprintf(`<pre>%s</pre>`, contents)
|
||||
case "md":
|
||||
switch r.TextMime {
|
||||
case "text/markdown":
|
||||
html := markdown.ToHTML(contents, nil, nil)
|
||||
ret += string(html)
|
||||
default:
|
||||
return "", errors.New("Unsupported markup: " + r.Markup)
|
||||
ret += fmt.Sprintf(`<pre>%s</pre>`, contents)
|
||||
}
|
||||
|
||||
ret += `
|
||||
</article>`
|
||||
return ret, nil
|
||||
}
|
||||
|
||||
func (r *Revision) ActionGetBinary(w http.ResponseWriter) {
|
||||
fileContents, err := ioutil.ReadFile(r.urlOfBinary())
|
||||
if err != nil {
|
||||
log.Println("Failed to load binary data of", r.FullName, r.Id)
|
||||
w.WriteHeader(http.StatusNotFound)
|
||||
return
|
||||
}
|
||||
w.Header().Set("Content-Type", r.BinaryMime)
|
||||
w.WriteHeader(http.StatusOK)
|
||||
w.Write(fileContents)
|
||||
log.Println("Serving binary data of", r.FullName, r.Id)
|
||||
}
|
||||
|
||||
func (r *Revision) ActionRaw(w http.ResponseWriter) {
|
||||
fileContents, err := ioutil.ReadFile(r.TextPath)
|
||||
if err != nil {
|
||||
log.Println("Failed to load text data of", r.FullName, r.Id)
|
||||
w.WriteHeader(http.StatusNotFound)
|
||||
return
|
||||
}
|
||||
w.Header().Set("Content-Type", r.TextMime)
|
||||
w.WriteHeader(http.StatusOK)
|
||||
w.Write(fileContents)
|
||||
log.Println("Serving text data of", r.FullName, r.Id)
|
||||
}
|
||||
|
||||
func (r *Revision) ActionZen(w http.ResponseWriter, hyphae map[string]*Hypha) {
|
||||
html, err := r.AsHtml(hyphae)
|
||||
if err != nil {
|
||||
log.Println("Failed to render", r.FullName)
|
||||
w.WriteHeader(http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
||||
w.WriteHeader(http.StatusOK)
|
||||
fmt.Fprint(w, html)
|
||||
}
|
||||
|
||||
func (r *Revision) ActionView(w http.ResponseWriter, hyphae map[string]*Hypha, layoutFun func(map[string]*Hypha, Revision, string) string) {
|
||||
html, err := r.AsHtml(hyphae)
|
||||
if err != nil {
|
||||
log.Println("Failed to render", r.FullName)
|
||||
w.WriteHeader(http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
||||
w.WriteHeader(http.StatusOK)
|
||||
fmt.Fprint(w, layoutFun(hyphae, *r, html))
|
||||
log.Println("Rendering", r.FullName)
|
||||
}
|
||||
func (r *Revision) Name() string {
|
||||
return r.FullName
|
||||
}
|
||||
|
@ -1,9 +0,0 @@
|
||||
{
|
||||
"createdAt": 1591636222,
|
||||
"comment": "update Fruit",
|
||||
"tags": ["fetus", "tasty"],
|
||||
"mimeType": "application/x-hypha",
|
||||
"markup": "md",
|
||||
"name": "Fruit",
|
||||
"author": "fungimaster"
|
||||
}
|
@ -1,8 +0,0 @@
|
||||
{
|
||||
"createdAt":1591635559,
|
||||
"comment":"create Fruit",
|
||||
"tags":["fetus", "tasty"],
|
||||
"mimeType":"application/x-hypha",
|
||||
"markup":"md",
|
||||
"name":"Fruit",
|
||||
"author":"bouncepaw"}
|
@ -1,3 +1,2 @@
|
||||
# Fruit
|
||||
Many people ask the question: what is fruit exactly?
|
||||
Fruit is a type of fetus. Is tasty so cool coo l ha i love fwriotwsn
|
@ -1,7 +0,0 @@
|
||||
{"createdAt":1591639464,
|
||||
"comment":"add apple pic hehehe",
|
||||
"tags":["img"],
|
||||
"mimeType":"image/jpeg",
|
||||
"markup":"plain",
|
||||
"name":"Apple",
|
||||
"author":"bouncepaw"}
|
Before Width: | Height: | Size: 43 KiB After Width: | Height: | Size: 43 KiB |
13
w/m/Fruit/Apple/meta.json
Normal file
13
w/m/Fruit/Apple/meta.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"revisions":{
|
||||
"1":{
|
||||
"name": "Apple",
|
||||
"time": 1591639464,
|
||||
"author": "bouncepaw",
|
||||
"comment": "add apple pic hehehe",
|
||||
"tags": ["img"],
|
||||
"text_mime": "text/plain",
|
||||
"binary_mime": "image/jpeg"
|
||||
}
|
||||
}
|
||||
}
|
@ -1 +0,0 @@
|
||||
{"creationTime": 1591639284}
|
20
w/m/Fruit/meta.json
Normal file
20
w/m/Fruit/meta.json
Normal file
@ -0,0 +1,20 @@
|
||||
{
|
||||
"revisions":{
|
||||
"1":{
|
||||
"name": "Fruit",
|
||||
"time": 1591635559,
|
||||
"author": "bouncepaw",
|
||||
"comment": "create Fruit",
|
||||
"tags": ["fetus", "tasty"],
|
||||
"text_mime": "text/markdown"
|
||||
},
|
||||
"2":{
|
||||
"name": "Fruit",
|
||||
"time": 1591636222,
|
||||
"author": "fungimaster",
|
||||
"comment": "update Fruit",
|
||||
"tags": ["fetus", "tasty"],
|
||||
"text_mime": "text/markdown"
|
||||
}
|
||||
}
|
||||
}
|
@ -1 +0,0 @@
|
||||
{"creationTime":1591635559}
|
2
w/m/sys/main.css/1.txt
Normal file
2
w/m/sys/main.css/1.txt
Normal file
@ -0,0 +1,2 @@
|
||||
b { color: red; }
|
||||
article { border: 1px black solid; }
|
11
w/m/sys/main.css/meta.json
Normal file
11
w/m/sys/main.css/meta.json
Normal file
@ -0,0 +1,11 @@
|
||||
{
|
||||
"revisions":{
|
||||
"1":{
|
||||
"name": "main.css",
|
||||
"time": 1592244023,
|
||||
"author": "wikimind",
|
||||
"comment": "make a placeholder style",
|
||||
"text_mime": "text/css"
|
||||
}
|
||||
}
|
||||
}
|
245
walk.go
245
walk.go
@ -2,190 +2,157 @@ package main
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"path/filepath"
|
||||
"regexp"
|
||||
"strconv"
|
||||
)
|
||||
|
||||
func scanHyphaDir(fullPath string) (structureMet bool, possibleRevisionPaths []string, possibleHyphaPaths []string, reterr error) {
|
||||
const (
|
||||
hyphaPattern = `[^\s\d:/?&\\][^:?&\\]*`
|
||||
hyphaUrl = `/{hypha:` + hyphaPattern + `}`
|
||||
revisionPattern = `[\d]+`
|
||||
revQuery = `{rev:` + revisionPattern + `}`
|
||||
revTxtPattern = revisionPattern + `\.txt`
|
||||
revBinPattern = revisionPattern + `\.bin`
|
||||
metaJsonPattern = `meta\.json`
|
||||
)
|
||||
|
||||
var (
|
||||
leadingInt = regexp.MustCompile(`^[-+]?\d+`)
|
||||
)
|
||||
|
||||
func matchNameToEverything(name string) (hyphaM bool, revTxtM bool, revBinM bool, metaJsonM bool) {
|
||||
simpleMatch := func(s string, p string) bool {
|
||||
m, _ := regexp.MatchString(p, s)
|
||||
return m
|
||||
}
|
||||
switch {
|
||||
case simpleMatch(name, revTxtPattern):
|
||||
revTxtM = true
|
||||
case simpleMatch(name, revBinPattern):
|
||||
revBinM = true
|
||||
case simpleMatch(name, metaJsonPattern):
|
||||
metaJsonM = true
|
||||
case simpleMatch(name, hyphaPattern):
|
||||
hyphaM = true
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func stripLeadingInt(s string) string {
|
||||
return leadingInt.FindString(s)
|
||||
}
|
||||
|
||||
func hyphaDirRevsValidate(dto map[string]map[string]string) (res bool) {
|
||||
for k, _ := range dto {
|
||||
switch k {
|
||||
case "0":
|
||||
delete(dto, "0")
|
||||
default:
|
||||
res = true
|
||||
}
|
||||
}
|
||||
return res
|
||||
}
|
||||
|
||||
func scanHyphaDir(fullPath string) (valid bool, revs map[string]map[string]string, possibleSubhyphae []string, metaJsonPath string, err error) {
|
||||
revs = make(map[string]map[string]string)
|
||||
nodes, err := ioutil.ReadDir(fullPath)
|
||||
if err != nil {
|
||||
reterr = err
|
||||
return // implicit return values
|
||||
}
|
||||
|
||||
var (
|
||||
mmJsonPresent bool
|
||||
zeroDirPresent bool
|
||||
)
|
||||
|
||||
for _, node := range nodes {
|
||||
matchedHypha, _ := regexp.MatchString(hyphaPattern, node.Name())
|
||||
matchedRev, _ := regexp.MatchString(revisionPattern, node.Name())
|
||||
hyphaM, revTxtM, revBinM, metaJsonM := matchNameToEverything(node.Name())
|
||||
switch {
|
||||
case matchedRev && node.IsDir():
|
||||
if node.Name() == "0" {
|
||||
zeroDirPresent = true
|
||||
case hyphaM && node.IsDir():
|
||||
possibleSubhyphae = append(possibleSubhyphae, filepath.Join(fullPath, node.Name()))
|
||||
case revTxtM && !node.IsDir():
|
||||
revId := stripLeadingInt(node.Name())
|
||||
if _, ok := revs[revId]; !ok {
|
||||
revs[revId] = make(map[string]string)
|
||||
}
|
||||
possibleRevisionPaths = append(
|
||||
possibleRevisionPaths,
|
||||
filepath.Join(fullPath, node.Name()),
|
||||
)
|
||||
case (node.Name() == "mm.json") && !node.IsDir():
|
||||
mmJsonPresent = true
|
||||
case matchedHypha && node.IsDir():
|
||||
possibleHyphaPaths = append(
|
||||
possibleHyphaPaths,
|
||||
filepath.Join(fullPath, node.Name()),
|
||||
)
|
||||
revs[revId]["txt"] = filepath.Join(fullPath, node.Name())
|
||||
case revBinM && !node.IsDir():
|
||||
revId := stripLeadingInt(node.Name())
|
||||
if _, ok := revs[revId]; !ok {
|
||||
revs[revId] = make(map[string]string)
|
||||
}
|
||||
revs[revId]["bin"] = filepath.Join(fullPath, node.Name())
|
||||
case metaJsonM && !node.IsDir():
|
||||
metaJsonPath = filepath.Join(fullPath, "meta.json")
|
||||
// Other nodes are ignored. It is not promised they will be ignored in future versions
|
||||
}
|
||||
}
|
||||
|
||||
if mmJsonPresent && zeroDirPresent {
|
||||
structureMet = true
|
||||
}
|
||||
valid = hyphaDirRevsValidate(revs)
|
||||
|
||||
return // implicit return values
|
||||
}
|
||||
|
||||
func check(e error) {
|
||||
if e != nil {
|
||||
panic(e)
|
||||
}
|
||||
}
|
||||
|
||||
// Hypha name is rootWikiDir/{here}
|
||||
func hyphaName(fullPath string) string {
|
||||
return fullPath[len(rootWikiDir)+1:]
|
||||
}
|
||||
|
||||
const (
|
||||
hyphaPattern = `[^\s\d:/?&\\][^:?&\\]*`
|
||||
revisionPattern = `[\d]+`
|
||||
hyphaUrl = "/{hypha:" + hyphaPattern + "}"
|
||||
revQuery = `{rev:[\d]+}`
|
||||
)
|
||||
|
||||
// Sends found hyphae to the `ch`. `fullPath` is tested for hyphaness, then its subdirs with hyphaesque names are tested too using goroutines for each subdir. The function is recursive.
|
||||
func recurFindHyphae(fullPath string) (hyphae []*Hypha) {
|
||||
|
||||
structureMet, possibleRevisionPaths, possibleHyphaPaths, err := scanHyphaDir(fullPath)
|
||||
func recurFindHyphae(fullPath string) map[string]*Hypha {
|
||||
hyphae := make(map[string]*Hypha)
|
||||
valid, revs, possibleSubhyphae, metaJsonPath, err := scanHyphaDir(fullPath)
|
||||
if err != nil {
|
||||
return hyphae
|
||||
}
|
||||
|
||||
// First, let's process inner hyphae
|
||||
for _, possibleHyphaPath := range possibleHyphaPaths {
|
||||
hyphae = append(hyphae, recurFindHyphae(possibleHyphaPath)...)
|
||||
// First, let's process subhyphae
|
||||
for _, possibleSubhypha := range possibleSubhyphae {
|
||||
for k, v := range recurFindHyphae(possibleSubhypha) {
|
||||
hyphae[k] = v
|
||||
}
|
||||
}
|
||||
|
||||
// This folder is not a hypha itself, nothing to do here
|
||||
if !structureMet {
|
||||
if !valid {
|
||||
return hyphae
|
||||
}
|
||||
|
||||
// Template hypha struct. Other fields are default jsont values.
|
||||
// Template hypha struct. Other fields are default json values.
|
||||
h := Hypha{
|
||||
FullName: hyphaName(fullPath),
|
||||
Path: fullPath,
|
||||
Name: hyphaName(fullPath),
|
||||
ParentName: filepath.Dir(hyphaName(fullPath)),
|
||||
parentName: filepath.Dir(hyphaName(fullPath)),
|
||||
// Children names are unknown now
|
||||
}
|
||||
|
||||
// Fill in every revision
|
||||
for _, possibleRevisionPath := range possibleRevisionPaths {
|
||||
rev, err := makeRevision(possibleRevisionPath, h.Name)
|
||||
if err == nil {
|
||||
h.Revisions = append(h.Revisions, rev)
|
||||
}
|
||||
metaJsonContents, err := ioutil.ReadFile(metaJsonPath)
|
||||
if err != nil {
|
||||
log.Printf("Error when reading `%s`; skipping", metaJsonPath)
|
||||
return hyphae
|
||||
}
|
||||
err = json.Unmarshal(metaJsonContents, &h)
|
||||
if err != nil {
|
||||
log.Printf("Error when unmarshaling `%s`; skipping", metaJsonPath)
|
||||
return hyphae
|
||||
}
|
||||
|
||||
mmJsonPath := filepath.Join(fullPath, "mm.json")
|
||||
mmJsonContents, err := ioutil.ReadFile(mmJsonPath)
|
||||
if err != nil {
|
||||
fmt.Println(fullPath, ">\tError:", err)
|
||||
return hyphae
|
||||
}
|
||||
err = json.Unmarshal(mmJsonContents, &h)
|
||||
if err != nil {
|
||||
fmt.Println(fullPath, ">\tError:", err)
|
||||
return hyphae
|
||||
// Fill in every revision paths
|
||||
for id, paths := range revs {
|
||||
if r, ok := h.Revisions[id]; ok {
|
||||
r.FullName = filepath.Join(h.parentName, r.ShortName)
|
||||
for fType, fPath := range paths {
|
||||
switch fType {
|
||||
case "bin":
|
||||
r.BinaryPath = fPath
|
||||
case "txt":
|
||||
r.TextPath = fPath
|
||||
}
|
||||
}
|
||||
} else {
|
||||
log.Printf("Error when reading hyphae from disk: hypha `%s`'s meta.json provided no information about revision `%s`, but files %s are provided; skipping\n", h.FullName, id, paths)
|
||||
}
|
||||
}
|
||||
|
||||
// Now the hypha should be ok, gotta send structs
|
||||
hyphae = append(hyphae, &h)
|
||||
hyphae[h.FullName] = &h
|
||||
return hyphae
|
||||
}
|
||||
|
||||
func makeRevision(fullPath string, fullName string) (r Revision, err error) {
|
||||
// fullPath is expected to be a path to a dir.
|
||||
// Revision directory must have at least `m.json` and `t.txt` files.
|
||||
var (
|
||||
mJsonPresent bool
|
||||
tTxtPresent bool
|
||||
bPresent bool
|
||||
)
|
||||
|
||||
nodes, err := ioutil.ReadDir(fullPath)
|
||||
if err != nil {
|
||||
return r, err
|
||||
}
|
||||
|
||||
for _, node := range nodes {
|
||||
if node.IsDir() {
|
||||
continue
|
||||
}
|
||||
switch node.Name() {
|
||||
case "m.json":
|
||||
mJsonPresent = true
|
||||
case "t.txt":
|
||||
tTxtPresent = true
|
||||
case "b":
|
||||
bPresent = true
|
||||
}
|
||||
}
|
||||
|
||||
if !(mJsonPresent && tTxtPresent) {
|
||||
return r, errors.New("makeRevision: m.json and t.txt files are not found")
|
||||
}
|
||||
|
||||
// If all the flags are true, this directory is assumed to be a revision. Gotta check further. This is template Revision struct. Other fields fall back to default init values.
|
||||
mJsonPath := filepath.Join(fullPath, "m.json")
|
||||
mJsonContents, err := ioutil.ReadFile(mJsonPath)
|
||||
if err != nil {
|
||||
fmt.Println(fullPath, ">\tError:", err)
|
||||
return r, err
|
||||
}
|
||||
|
||||
r = Revision{FullName: fullName}
|
||||
err = json.Unmarshal(mJsonContents, &r)
|
||||
if err != nil {
|
||||
fmt.Println(fullPath, ">\tError:", err)
|
||||
return r, err
|
||||
}
|
||||
|
||||
// Now, let's fill in t.txt path
|
||||
r.TextPath = filepath.Join(fullPath, "t.txt")
|
||||
|
||||
// There's sense in reading binary file only if the hypha is marked as such
|
||||
if r.MimeType != "application/x-hypha" {
|
||||
// Do not check for binary file presence, attempt to read it will fail anyway
|
||||
if bPresent {
|
||||
r.BinaryPath = filepath.Join(fullPath, "b")
|
||||
r.BinaryRequest = fmt.Sprintf("%s?rev=%d&action=getBinary", r.FullName, r.Id)
|
||||
} else {
|
||||
return r, errors.New("makeRevision: b file not present")
|
||||
}
|
||||
}
|
||||
|
||||
// So far, so good. Let's fill in id. It is guaranteed to be correct, so no error checking
|
||||
id, _ := strconv.Atoi(filepath.Base(fullPath))
|
||||
r.Id = id
|
||||
|
||||
// It is safe now to return, I guess
|
||||
return r, nil
|
||||
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user