filebrowser/editor.go

165 lines
3.7 KiB
Go
Raw Normal View History

2017-06-24 11:12:15 +00:00
package filemanager
import (
"bytes"
"errors"
"net/http"
"path/filepath"
"strings"
"github.com/hacdias/filemanager/frontmatter"
"github.com/spf13/hugo/parser"
)
2017-06-25 13:24:26 +00:00
// editor contains the information to fill the editor template.
type editor struct {
2017-06-24 11:12:15 +00:00
Class string
Mode string
Visual bool
Content string
FrontMatter struct {
Content *frontmatter.Content
Rune rune
}
}
2017-06-25 13:24:26 +00:00
// getEditor gets the editor based on a Info struct
2017-06-25 13:48:34 +00:00
func getEditor(r *http.Request, i *fileInfo) (*editor, error) {
2017-06-24 11:12:15 +00:00
var err error
// Create a new editor variable and set the mode
2017-06-25 13:24:26 +00:00
e := &editor{}
2017-06-24 11:12:15 +00:00
e.Mode = editorMode(i.Name)
e.Class = editorClass(e.Mode)
if e.Class == "frontmatter-only" || e.Class == "complete" {
e.Visual = true
}
if r.URL.Query().Get("visual") == "false" {
e.Class = "content-only"
}
hasRune := frontmatter.HasRune(i.content)
if e.Class == "frontmatter-only" && !hasRune {
e.FrontMatter.Rune, err = frontmatter.StringFormatToRune(e.Mode)
if err != nil {
goto Error
}
i.content = frontmatter.AppendRune(i.content, e.FrontMatter.Rune)
hasRune = true
}
if e.Class == "frontmatter-only" && hasRune {
e.FrontMatter.Content, _, err = frontmatter.Pretty(i.content)
if err != nil {
goto Error
}
}
if e.Class == "complete" && hasRune {
var page parser.Page
// Starts a new buffer and parses the file using Hugo's functions
buffer := bytes.NewBuffer(i.content)
page, err = parser.ReadFrom(buffer)
if err != nil {
goto Error
}
// Parses the page content and the frontmatter
e.Content = strings.TrimSpace(string(page.Content()))
e.FrontMatter.Rune = rune(i.content[0])
e.FrontMatter.Content, _, err = frontmatter.Pretty(page.FrontMatter())
}
if e.Class == "complete" && !hasRune {
err = errors.New("Complete but without rune")
}
Error:
if e.Class == "content-only" || err != nil {
e.Class = "content-only"
e.Content = i.StringifyContent()
}
return e, nil
}
2017-06-27 08:28:29 +00:00
// serveSingle serves a single file in an editor (if it is editable), shows the
// plain file, or downloads it if it can't be shown.
2017-06-27 08:57:11 +00:00
func serveSingle(ctx *requestContext, w http.ResponseWriter, r *http.Request) (int, error) {
2017-06-27 08:28:29 +00:00
var err error
2017-06-27 08:57:11 +00:00
if err = ctx.Info.RetrieveFileType(); err != nil {
2017-06-27 08:28:29 +00:00
return errorToHTTP(err, true), err
}
p := &page{
2017-06-27 08:57:11 +00:00
Name: ctx.Info.Name,
Path: ctx.Info.VirtualPath,
2017-06-27 08:28:29 +00:00
IsDir: false,
2017-06-27 08:57:11 +00:00
Data: ctx.Info,
User: ctx.User,
PrefixURL: ctx.FileManager.PrefixURL,
BaseURL: ctx.FileManager.AbsoluteURL(),
WebDavURL: ctx.FileManager.AbsoluteWebDavURL(),
2017-06-27 08:28:29 +00:00
}
// If the request accepts JSON, we send the file information.
if strings.Contains(r.Header.Get("Accept"), "application/json") {
return p.PrintAsJSON(w)
}
2017-06-27 08:57:11 +00:00
if ctx.Info.Type == "text" {
if err = ctx.Info.Read(); err != nil {
2017-06-27 08:28:29 +00:00
return errorToHTTP(err, true), err
}
}
2017-06-27 08:57:11 +00:00
if ctx.Info.CanBeEdited() && ctx.User.AllowEdit {
p.Data, err = getEditor(r, ctx.Info)
2017-06-27 08:28:29 +00:00
p.Editor = true
if err != nil {
return http.StatusInternalServerError, err
}
2017-06-27 08:57:11 +00:00
return p.PrintAsHTML(w, ctx.FileManager.assets.templates, "frontmatter", "editor")
2017-06-27 08:28:29 +00:00
}
2017-06-27 08:57:11 +00:00
return p.PrintAsHTML(w, ctx.FileManager.assets.templates, "single")
2017-06-27 08:28:29 +00:00
}
2017-06-24 11:12:15 +00:00
func editorClass(mode string) string {
switch mode {
case "json", "toml", "yaml":
return "frontmatter-only"
case "markdown", "asciidoc", "rst":
return "complete"
}
return "content-only"
}
func editorMode(filename string) string {
mode := strings.TrimPrefix(filepath.Ext(filename), ".")
switch mode {
case "md", "markdown", "mdown", "mmark":
mode = "markdown"
case "asciidoc", "adoc", "ad":
mode = "asciidoc"
case "rst":
mode = "rst"
case "html", "htm":
mode = "html"
case "js":
mode = "javascript"
case "go":
mode = "golang"
}
return mode
}