general improvements; updates on #32
This commit is contained in:
parent
44065cfaf9
commit
f5660fbc48
|
@ -3,6 +3,7 @@
|
||||||
[](https://travis-ci.org/hacdias/caddy-filemanager)
|
[](https://travis-ci.org/hacdias/caddy-filemanager)
|
||||||
[](https://forum.caddyserver.com)
|
[](https://forum.caddyserver.com)
|
||||||
[](http://godoc.org/github.com/hacdias/caddy-filemanager)
|
[](http://godoc.org/github.com/hacdias/caddy-filemanager)
|
||||||
|
[](https://goreportcard.com/report/hacdias/caddy-filemanager
|
||||||
|
|
||||||
This package is a plugin for Caddy server that provides an online file manager (based on browse middleware) that is able to: rename files, delete files and upload files. Some new features that can be implemented in the future can be seen at [issues](https://github.com/hacdias/caddy-filemanager/issues).
|
This package is a plugin for Caddy server that provides an online file manager (based on browse middleware) that is able to: rename files, delete files and upload files. Some new features that can be implemented in the future can be seen at [issues](https://github.com/hacdias/caddy-filemanager/issues).
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,7 @@ import (
|
||||||
"github.com/mholt/caddy/caddyhttp/httpserver"
|
"github.com/mholt/caddy/caddyhttp/httpserver"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Config is a configuration for browsing in a particualr path.
|
// Config is a configuration for browsing in a particular path.
|
||||||
type Config struct {
|
type Config struct {
|
||||||
*User
|
*User
|
||||||
BaseURL string
|
BaseURL string
|
||||||
|
@ -66,7 +66,7 @@ func Parse(c *caddy.Controller) ([]Config, error) {
|
||||||
cfg.AllowEdit = true
|
cfg.AllowEdit = true
|
||||||
cfg.AllowNew = true
|
cfg.AllowNew = true
|
||||||
cfg.Commands = []string{"git", "svn", "hg"}
|
cfg.Commands = []string{"git", "svn", "hg"}
|
||||||
cfg.Rules = []*Rule{&Rule{
|
cfg.Rules = []*Rule{{
|
||||||
Regex: true,
|
Regex: true,
|
||||||
Allow: false,
|
Allow: false,
|
||||||
Regexp: regexp.MustCompile("\\/\\..+"),
|
Regexp: regexp.MustCompile("\\/\\..+"),
|
||||||
|
|
|
@ -2,7 +2,6 @@ package handlers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
@ -18,167 +17,120 @@ var upgrader = websocket.Upgrader{
|
||||||
WriteBufferSize: 1024,
|
WriteBufferSize: 1024,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
cmdNotImplemented = []byte("Command not implemented.")
|
||||||
|
cmdNotAllowed = []byte("Command not allowed.")
|
||||||
|
)
|
||||||
|
|
||||||
// Command handles the requests for VCS related commands: git, svn and mercurial
|
// Command handles the requests for VCS related commands: git, svn and mercurial
|
||||||
func Command(w http.ResponseWriter, r *http.Request, c *config.Config, u *config.User) (int, error) {
|
func Command(w http.ResponseWriter, r *http.Request, c *config.Config, u *config.User) (int, error) {
|
||||||
|
// Upgrades the connection to a websocket and checks for errors.
|
||||||
conn, err := upgrader.Upgrade(w, r, nil)
|
conn, err := upgrader.Upgrade(w, r, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
return 0, err
|
||||||
return 0, nil
|
|
||||||
}
|
}
|
||||||
defer conn.Close()
|
defer conn.Close()
|
||||||
|
|
||||||
|
var (
|
||||||
|
message []byte
|
||||||
|
command []string
|
||||||
|
)
|
||||||
|
|
||||||
|
// Starts an infinite loop until a valid command is captured.
|
||||||
for {
|
for {
|
||||||
_, message, err := conn.ReadMessage()
|
_, message, err = conn.ReadMessage()
|
||||||
if err != nil {
|
|
||||||
fmt.Println("read:", err)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
command := strings.Split(string(message), " ")
|
|
||||||
|
|
||||||
if len(command) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
// Check if the command is allowed
|
|
||||||
mayContinue := false
|
|
||||||
|
|
||||||
for _, cmd := range u.Commands {
|
|
||||||
if cmd == command[0] {
|
|
||||||
mayContinue = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if !mayContinue {
|
|
||||||
err = conn.WriteMessage(websocket.BinaryMessage, []byte("FORBIDDEN"))
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("write:", err)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if the program is talled is installed on the computer
|
|
||||||
if _, err = exec.LookPath(command[0]); err != nil {
|
|
||||||
err = conn.WriteMessage(websocket.BinaryMessage, []byte("Command not implemented."))
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("write:", err)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
return http.StatusNotImplemented, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
path := strings.Replace(r.URL.Path, c.BaseURL, c.Scope, 1)
|
|
||||||
path = filepath.Clean(path)
|
|
||||||
|
|
||||||
buff := new(bytes.Buffer)
|
|
||||||
|
|
||||||
cmd := exec.Command(command[0], command[1:len(command)]...)
|
|
||||||
cmd.Dir = path
|
|
||||||
cmd.Stderr = buff
|
|
||||||
cmd.Stdout = buff
|
|
||||||
err = cmd.Start()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return http.StatusInternalServerError, err
|
return http.StatusInternalServerError, err
|
||||||
}
|
}
|
||||||
|
|
||||||
done := false
|
command = strings.Split(string(message), " ")
|
||||||
go func() {
|
if len(command) != 0 {
|
||||||
err = cmd.Wait()
|
break
|
||||||
done = true
|
|
||||||
}()
|
|
||||||
|
|
||||||
for !done {
|
|
||||||
by := buff.Bytes()
|
|
||||||
if len(by) > 0 {
|
|
||||||
err = conn.WriteMessage(websocket.TextMessage, by)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("write:", err)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
time.Sleep(100 * time.Millisecond)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
by := buff.Bytes()
|
|
||||||
if len(by) > 0 {
|
|
||||||
err = conn.WriteMessage(websocket.TextMessage, by)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("write:", err)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
time.Sleep(100 * time.Millisecond)
|
|
||||||
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* command := strings.Split(r.Header.Get("command"), " ")
|
|
||||||
|
|
||||||
// Check if the command is allowed
|
// Check if the command is allowed
|
||||||
mayContinue := false
|
allowed := false
|
||||||
|
|
||||||
for _, cmd := range u.Commands {
|
for _, cmd := range u.Commands {
|
||||||
if cmd == command[0] {
|
if cmd == command[0] {
|
||||||
mayContinue = true
|
allowed = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !mayContinue {
|
if !allowed {
|
||||||
return http.StatusForbidden, nil
|
err = conn.WriteMessage(websocket.BinaryMessage, cmdNotAllowed)
|
||||||
|
if err != nil {
|
||||||
|
return http.StatusInternalServerError, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if the program is talled is installed on the computer
|
// Check if the program is talled is installed on the computer.
|
||||||
if _, err := exec.LookPath(command[0]); err != nil {
|
if _, err = exec.LookPath(command[0]); err != nil {
|
||||||
|
err = conn.WriteMessage(websocket.BinaryMessage, cmdNotImplemented)
|
||||||
|
if err != nil {
|
||||||
|
return http.StatusInternalServerError, err
|
||||||
|
}
|
||||||
|
|
||||||
return http.StatusNotImplemented, nil
|
return http.StatusNotImplemented, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Gets the path and initializes a buffer.
|
||||||
path := strings.Replace(r.URL.Path, c.BaseURL, c.Scope, 1)
|
path := strings.Replace(r.URL.Path, c.BaseURL, c.Scope, 1)
|
||||||
path = filepath.Clean(path)
|
path = filepath.Clean(path)
|
||||||
|
buff := new(bytes.Buffer)
|
||||||
|
|
||||||
cmd := exec.Command(command[0], command[1:len(command)]...)
|
// Sets up the command executation.
|
||||||
|
cmd := exec.Command(command[0], command[1:]...)
|
||||||
cmd.Dir = path
|
cmd.Dir = path
|
||||||
cmd.Stderr = w
|
cmd.Stderr = buff
|
||||||
cmd.Stdout = w
|
cmd.Stdout = buff
|
||||||
cmd.Start()
|
|
||||||
|
|
||||||
/*cmd.Stderr = b
|
// Starts the command and checks for errors.
|
||||||
cmd.Stdout = b
|
err = cmd.Start()
|
||||||
|
|
||||||
// Starts the comamnd
|
|
||||||
err := cmd.Start()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return http.StatusInternalServerError, err
|
return http.StatusInternalServerError, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Set a 'done' variable to check whetever the command has already finished
|
||||||
|
// running or not. This verification is done using a goroutine that uses the
|
||||||
|
// method .Wait() from the command.
|
||||||
done := false
|
done := false
|
||||||
go func() {
|
go func() {
|
||||||
err = cmd.Wait()
|
err = cmd.Wait()
|
||||||
done = true
|
done = true
|
||||||
}()
|
}()
|
||||||
|
|
||||||
for !done {
|
// Function to print the current information on the buffer to the connection.
|
||||||
by := b.Bytes()
|
print := func() error {
|
||||||
|
by := buff.Bytes()
|
||||||
if len(by) > 0 {
|
if len(by) > 0 {
|
||||||
fmt.Println(string(by))
|
err = conn.WriteMessage(websocket.TextMessage, by)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//w.Write(by)
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
}*/
|
// While the command hasn't finished running, continue sending the output
|
||||||
|
// to the client in intervals of 100 milliseconds.
|
||||||
|
for !done {
|
||||||
|
if err = print(); err != nil {
|
||||||
|
return http.StatusInternalServerError, err
|
||||||
|
}
|
||||||
|
|
||||||
//out, err := cmd.CombinedOutput()
|
time.Sleep(100 * time.Millisecond)
|
||||||
//fmt.Println(string(out))
|
}
|
||||||
|
|
||||||
//if err != nil {
|
// After the command is done executing, send the output one more time to the
|
||||||
// return http.StatusInternalServerError, err
|
// browser to make sure it gets the latest information.
|
||||||
//}
|
if err = print(); err != nil {
|
||||||
|
return http.StatusInternalServerError, err
|
||||||
|
}
|
||||||
|
|
||||||
/* cmd.Wait()
|
|
||||||
|
|
||||||
//p := &page.Page{Info: &page.Info{Data: string(output)}} */
|
|
||||||
return 0, nil
|
return 0, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -70,7 +70,10 @@ func GetEditor(i *file.Info) (*Editor, error) {
|
||||||
// Parses the page content and the frontmatter
|
// Parses the page content and the frontmatter
|
||||||
editor.Content = strings.TrimSpace(string(page.Content()))
|
editor.Content = strings.TrimSpace(string(page.Content()))
|
||||||
editor.FrontMatter, _, err = frontmatter.Pretty(page.FrontMatter())
|
editor.FrontMatter, _, err = frontmatter.Pretty(page.FrontMatter())
|
||||||
break
|
|
||||||
|
if err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -133,7 +133,7 @@ func (p Page) PrintAsHTML(w http.ResponseWriter, templates ...string) (int, erro
|
||||||
|
|
||||||
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
||||||
_, err = buf.WriteTo(w)
|
_, err = buf.WriteTo(w)
|
||||||
return http.StatusOK, nil
|
return http.StatusOK, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// PrintAsJSON prints the current Page infromation in JSON
|
// PrintAsJSON prints the current Page infromation in JSON
|
||||||
|
|
Loading…
Reference in New Issue