package fs import ( "encoding/json" "errors" "io/ioutil" "log" "net/http" "os" "path/filepath" "strconv" "strings" "time" "github.com/bouncepaw/mycorrhiza/mycelium" "github.com/bouncepaw/mycorrhiza/util" ) type Hypha struct { Exists bool `json:"-"` FullName string `json:"-"` ViewCount int `json:"views"` Deleted bool `json:"deleted"` Revisions map[string]*Revision `json:"revisions"` actual *Revision `json:"-"` Invalid bool `json:"-"` Err error `json:"-"` } func (h *Hypha) Invalidate(err error) *Hypha { h.Invalid = true h.Err = err return h } func (s *Storage) OpenFromMap(m map[string]string) *Hypha { name := mycelium.NameWithMyceliumInMap(m) h := s.open(name) if rev, ok := m["rev"]; ok { return h.OnRevision(rev) } return h } func (s *Storage) open(name string) *Hypha { name = util.UrlToCanonical(name) h := &Hypha{ Exists: true, FullName: util.CanonicalToDisplay(name), } path, ok := s.paths[name] // This hypha does not exist yet if !ok { log.Println("Hypha", name, "does not exist") h.Exists = false h.Revisions = make(map[string]*Revision) } else { metaJsonText, err := ioutil.ReadFile(filepath.Join(path, "meta.json")) if err != nil { return h.Invalidate(err) } err = json.Unmarshal(metaJsonText, &h) if err != nil { return h.Invalidate(err) } // fill in rooted paths to content files and full names for idStr, rev := range h.Revisions { rev.FullName = filepath.Join(h.parentName(), rev.ShortName) rev.Id, _ = strconv.Atoi(idStr) if rev.BinaryName != "" { rev.BinaryPath = filepath.Join(path, rev.BinaryName) } rev.TextPath = filepath.Join(path, rev.TextName) } return h.OnRevision("0") } return h } // OnRevision tries to change to a revision specified by `id`. func (h *Hypha) OnRevision(id string) *Hypha { if h.Invalid || !h.Exists { return h } if len(h.Revisions) == 0 { return h.Invalidate(errors.New("This hypha has no revisions")) } if id == "0" { id = h.NewestId() } // Revision must be there, so no error checking if rev, _ := h.Revisions[id]; true { h.actual = rev } return h } func (h *Hypha) PlainLog(s string) { if h.Exists { log.Println(h.FullName, h.actual.Id, s) } else { log.Println("nonexistent", h.FullName, s) } } func (h *Hypha) LogSuccMaybe(succMsg string) *Hypha { if h.Invalid { h.PlainLog(h.Err.Error()) } else { h.PlainLog(succMsg) } return h } // ActionRaw is used with `?action=raw`. // It writes text content of the revision without any parsing or rendering. func (h *Hypha) ActionRaw(w http.ResponseWriter) *Hypha { if h.Invalid { return h } if h.Exists { fileContents, err := ioutil.ReadFile(h.actual.TextPath) if err != nil { return h.Invalidate(err) } w.Header().Set("Content-Type", h.mimeTypeForActionRaw()) w.WriteHeader(http.StatusOK) w.Write(fileContents) } else { log.Println("Hypha", h.FullName, "has no actual revision") w.WriteHeader(http.StatusNotFound) } return h } // ActionBinary is used with `?action=binary`. // It writes contents of binary content file. func (h *Hypha) ActionBinary(w http.ResponseWriter) *Hypha { if h.Invalid { return h } if h.Exists { fileContents, err := ioutil.ReadFile(h.actual.BinaryPath) if err != nil { return h.Invalidate(err) } w.Header().Set("Content-Type", h.actual.BinaryMime) w.WriteHeader(http.StatusOK) w.Write(fileContents) } else { log.Println("Hypha", h.FullName, "has no actual revision") w.WriteHeader(http.StatusNotFound) } return h } // ActionZen is used with `?action=zen`. // It renders the hypha but without any layout or styles. Pure. Zen. func (h *Hypha) ActionZen(w http.ResponseWriter) *Hypha { if h.Invalid { return h } html, err := h.asHtml() if err != nil { w.WriteHeader(http.StatusInternalServerError) return h.Invalidate(err) } w.Header().Set("Content-Type", "text/html;charset=utf-8") w.WriteHeader(http.StatusOK) w.Write([]byte(html)) return h } // ActionView is used with `?action=view` or no action at all. // It renders the page, the layout and everything else. func (h *Hypha) ActionView(w http.ResponseWriter, renderExists, renderNotExists func(string, string) []byte) *Hypha { var html string var err error if h.Exists { html, err = h.asHtml() if err != nil { w.WriteHeader(http.StatusInternalServerError) return h.Invalidate(err) } } w.Header().Set("Content-Type", "text/html;charset=utf-8") w.WriteHeader(http.StatusOK) if h.Exists { w.Write(renderExists(h.FullName, html)) } else { w.Write(renderNotExists(h.FullName, "")) } return h } // CreateDirIfNeeded creates directory where the hypha must reside if needed. // It is not needed if the dir already exists. func (h *Hypha) CreateDirIfNeeded() *Hypha { if h.Invalid { return h } // os.MkdirAll created dir if it is not there. Basically, checks it for us. err := os.MkdirAll(h.Path(), os.ModePerm) if err != nil { h.Invalidate(err) } return h } // makeTagsSlice turns strings like `"foo,, bar,kek"` to slice of strings that represent tag names. Whitespace around commas is insignificant. // Expected output for string above: []string{"foo", "bar", "kek"} func makeTagsSlice(responseTagsString string) (ret []string) { for _, tag := range strings.Split(responseTagsString, ",") { if trimmed := strings.TrimSpace(tag); "" == trimmed { ret = append(ret, trimmed) } } return ret } // revisionFromHttpData creates a new revison for hypha `h`. All data is fetched from `rq`, except for BinaryMime and BinaryPath which require additional processing. The revision is inserted for you. You'll have to pop it out if there is an error. func (h *Hypha) AddRevisionFromHttpData(rq *http.Request) *Hypha { if h.Invalid { return h } id := 1 if h.Exists { id = h.actual.Id + 1 } log.Printf("Creating revision %d from http data", id) rev := &Revision{ Id: id, FullName: h.FullName, ShortName: filepath.Base(h.FullName), Tags: makeTagsSlice(rq.PostFormValue("tags")), Comment: rq.PostFormValue("comment"), Author: rq.PostFormValue("author"), Time: int(time.Now().Unix()), TextMime: rq.PostFormValue("text_mime"), // Fields left: BinaryMime, BinaryPath, BinaryName, TextName, TextPath } rev.generateTextFilename() // TextName is set now rev.TextPath = filepath.Join(h.Path(), rev.TextName) return h.AddRevision(rev) } func (h *Hypha) AddRevision(rev *Revision) *Hypha { if h.Invalid { return h } h.Revisions[strconv.Itoa(rev.Id)] = rev h.actual = rev return h } // WriteTextFileFromHttpData tries to fetch text content from `rq` for revision `rev` and write it to a corresponding text file. It used in `HandlerUpdate`. func (h *Hypha) WriteTextFileFromHttpData(rq *http.Request) *Hypha { if h.Invalid { return h } data := []byte(rq.PostFormValue("text")) err := ioutil.WriteFile(h.TextPath(), data, 0644) if err != nil { log.Println("Failed to write", len(data), "bytes to", h.TextPath()) h.Invalidate(err) } return h } // WriteBinaryFileFromHttpData tries to fetch binary content from `rq` for revision `newRev` and write it to a corresponding binary file. If there is no content, it is taken from a previous revision, if there is any. func (h *Hypha) WriteBinaryFileFromHttpData(rq *http.Request) *Hypha { if h.Invalid { return h } // 10 MB file size limit rq.ParseMultipartForm(10 << 20) // Read file file, handler, err := rq.FormFile("binary") if file != nil { defer file.Close() } // If file is not passed: if err != nil { // Let's hope there are no other errors 🙏 // TODO: actually check if there any other errors log.Println("No binary data passed for", h.FullName) // It is expected there is at least one revision if len(h.Revisions) > 1 { prevRev := h.Revisions[strconv.Itoa(h.actual.Id-1)] h.actual.BinaryMime = prevRev.BinaryMime h.actual.BinaryPath = prevRev.BinaryPath h.actual.BinaryName = prevRev.BinaryName log.Println("Set previous revision's binary data") } return h } // If file is passed: h.actual.BinaryMime = handler.Header.Get("Content-Type") h.actual.generateBinaryFilename() h.actual.BinaryPath = filepath.Join(h.Path(), h.actual.BinaryName) data, err := ioutil.ReadAll(file) if err != nil { return h.Invalidate(err) } log.Println("Got", len(data), "of binary data for", h.FullName) err = ioutil.WriteFile(h.actual.BinaryPath, data, 0644) if err != nil { return h.Invalidate(err) } log.Println("Written", len(data), "of binary data for", h.FullName) return h } // SaveJson dumps the hypha's metadata to `meta.json` file. func (h *Hypha) SaveJson() *Hypha { if h.Invalid { return h } data, err := json.MarshalIndent(h, "", "\t") if err != nil { return h.Invalidate(err) } err = ioutil.WriteFile(h.MetaJsonPath(), data, 0644) if err != nil { return h.Invalidate(err) } log.Println("Saved JSON data of", h.FullName) return h } // Store adds `h` to the `Hs` if it is not already there func (h *Hypha) Store() *Hypha { if !h.Invalid { Hs.paths[h.CanonicalName()] = h.Path() } return h }