better error pages

pull/68/head
Henrique Dias 2016-03-12 13:03:31 +00:00
parent 012cd383a0
commit b57bcb6b0e
13 changed files with 112 additions and 91 deletions

File diff suppressed because one or more lines are too long

View File

@ -186,7 +186,7 @@ $(document).on('page:browse', function() {
if (request.status == 200) { if (request.status == 200) {
$.pjax({ $.pjax({
url: request.Location, url: response.location,
container: '#content' container: '#content'
}) })
} }

View File

@ -0,0 +1,8 @@
{{ define "content" }}
<main role="main" class="container browse error">
<h1>{{ .Title }}</h1>
<code>{{ .Message }}</code>
<p><strong>If this error persists contact us on <a href="https://gitter.im/hacdias/caddy-hugo">Gitter chat</a> or open an issue at <a href="https://github.com/hacdias/caddy-hugo/issues/new">GitHub</a>.</strong></p>
</main>
{{ end }}

View File

@ -18,6 +18,7 @@ import (
"github.com/hacdias/caddy-hugo/routes/assets" "github.com/hacdias/caddy-hugo/routes/assets"
"github.com/hacdias/caddy-hugo/routes/browse" "github.com/hacdias/caddy-hugo/routes/browse"
"github.com/hacdias/caddy-hugo/routes/editor" "github.com/hacdias/caddy-hugo/routes/editor"
"github.com/hacdias/caddy-hugo/routes/errors"
"github.com/hacdias/caddy-hugo/routes/git" "github.com/hacdias/caddy-hugo/routes/git"
"github.com/hacdias/caddy-hugo/tools/commands" "github.com/hacdias/caddy-hugo/tools/commands"
"github.com/hacdias/caddy-hugo/tools/hugo" "github.com/hacdias/caddy-hugo/tools/hugo"
@ -143,6 +144,10 @@ func (h CaddyHugo) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error
code, err = git.ServeHTTP(w, r, h.Config) code, err = git.ServeHTTP(w, r, h.Config)
} }
if code != 0 && code != 200 {
code, err = errors.ServeHTTP(w, r, code, err)
}
return code, err return code, err
} }

View File

@ -10,6 +10,11 @@ import (
var conf *config.Config var conf *config.Config
type response struct {
Message string `json:"message"`
Location string `json:"location"`
}
// ServeHTTP is used to serve the content of Browse page using Browse middleware // ServeHTTP is used to serve the content of Browse page using Browse middleware
// from Caddy. It handles the requests for DELETE, POST, GET and PUT related to // from Caddy. It handles the requests for DELETE, POST, GET and PUT related to
// /browse interface. // /browse interface.

View File

@ -5,7 +5,7 @@ import (
"os" "os"
"strings" "strings"
"github.com/hacdias/caddy-hugo/tools/server" s "github.com/hacdias/caddy-hugo/tools/server"
) )
// DELETE handles the delete requests on browse pages // DELETE handles the delete requests on browse pages
@ -31,17 +31,11 @@ func DELETE(w http.ResponseWriter, r *http.Request) (int, error) {
// Check for errors // Check for errors
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{err.Error(), ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, nil)
} }
} else { } else {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"File not found.", ""}, http.StatusNotFound, nil)
"message": "File not found.",
}, 404, nil)
} }
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{message, ""}, http.StatusOK, nil)
"message": message,
}, 200, nil)
} }

View File

@ -11,7 +11,7 @@ import (
"strings" "strings"
"github.com/hacdias/caddy-hugo/tools/commands" "github.com/hacdias/caddy-hugo/tools/commands"
"github.com/hacdias/caddy-hugo/tools/server" s "github.com/hacdias/caddy-hugo/tools/server"
) )
// POST handles the POST method on browse page. It's used to create new files, // POST handles the POST method on browse page. It's used to create new files,
@ -36,15 +36,11 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
// Check if filename and archetype are specified in // Check if filename and archetype are specified in
// the request // the request
if _, ok := info["filename"]; !ok { if _, ok := info["filename"]; !ok {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Filename not specified.", ""}, http.StatusBadRequest, nil)
"message": "Filename not specified.",
}, 500, nil)
} }
if _, ok := info["archetype"]; !ok { if _, ok := info["archetype"]; !ok {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Archtype not specified.", ""}, http.StatusBadRequest, nil)
"message": "Archtype not specified.",
}, 500, nil)
} }
// Sanitize the file name path // Sanitize the file name path
@ -66,9 +62,7 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
} }
if err := commands.Run(conf.Hugo, args, conf.Path); err != nil { if err := commands.Run(conf.Hugo, args, conf.Path); err != nil {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Something went wrong.", ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, err)
} }
} else { } else {
var err error var err error
@ -83,26 +77,19 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
} }
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Something went wrong.", ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, err)
} }
} }
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"File created!", url}, http.StatusOK, nil)
"location": url,
"message": "File created.",
}, 200, nil)
} }
func upload(w http.ResponseWriter, r *http.Request) (int, error) { func upload(w http.ResponseWriter, r *http.Request) (int, error) {
// Parse the multipart form in the request // Parse the multipart form in the request
err := r.ParseMultipartForm(100000) err := r.ParseMultipartForm(100000)
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Something went wrong.", ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, err)
} }
// For each file header in the multipart form // For each file header in the multipart form
@ -112,29 +99,23 @@ func upload(w http.ResponseWriter, r *http.Request) (int, error) {
// Open the first file // Open the first file
var infile multipart.File var infile multipart.File
if infile, err = hdr.Open(); nil != err { if infile, err = hdr.Open(); nil != err {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Something went wrong.", ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, err)
} }
// Create the file // Create the file
var outfile *os.File var outfile *os.File
if outfile, err = os.Create(conf.Path + r.URL.Path + hdr.Filename); nil != err { if outfile, err = os.Create(conf.Path + r.URL.Path + hdr.Filename); nil != err {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Something went wrong.", ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, err)
} }
// Copy the file content // Copy the file content
if _, err = io.Copy(outfile, infile); nil != err { if _, err = io.Copy(outfile, infile); nil != err {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &response{"Something went wrong.", ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, err)
} }
defer outfile.Close() defer outfile.Close()
} }
} }
return server.RespondJSON(w, nil, 200, nil) return s.RespondJSON(w, nil, http.StatusOK, nil)
} }

View File

@ -30,9 +30,7 @@ func PUT(w http.ResponseWriter, r *http.Request) (int, error) {
// Check if filename and archetype are specified in // Check if filename and archetype are specified in
// the request // the request
if _, ok := info["filename"]; !ok { if _, ok := info["filename"]; !ok {
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{"Filename not specified.", ""}, http.StatusBadRequest, nil)
"message": "Filename not specified.",
}, 400, nil)
} }
// Sanitize the file name path // Sanitize the file name path
@ -43,12 +41,8 @@ func PUT(w http.ResponseWriter, r *http.Request) (int, error) {
// Renames the file/folder // Renames the file/folder
if err := os.Rename(old, new); err != nil { if err := os.Rename(old, new); err != nil {
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{err.Error(), ""}, http.StatusInternalServerError, err)
"message": "Something went wrong.",
}, 500, err)
} }
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{"File renamed.", ""}, http.StatusOK, nil)
"message": "File renamed.",
}, 200, nil)
} }

View File

@ -37,9 +37,9 @@ func GET(w http.ResponseWriter, r *http.Request) (int, error) {
// Check if the file exists. // Check if the file exists.
if _, err := os.Stat(filename); os.IsNotExist(err) { if _, err := os.Stat(filename); os.IsNotExist(err) {
return http.StatusNotFound, nil return http.StatusNotFound, err
} else if os.IsPermission(err) { } else if os.IsPermission(err) {
return http.StatusForbidden, nil return http.StatusForbidden, err
} else if err != nil { } else if err != nil {
return http.StatusInternalServerError, err return http.StatusInternalServerError, err
} }
@ -47,7 +47,7 @@ func GET(w http.ResponseWriter, r *http.Request) (int, error) {
// Open the file and check if there was some error while opening // Open the file and check if there was some error while opening
file, err := ioutil.ReadFile(filename) file, err := ioutil.ReadFile(filename)
if os.IsPermission(err) { if os.IsPermission(err) {
return http.StatusForbidden, nil return http.StatusForbidden, err
} else if err != nil { } else if err != nil {
return http.StatusInternalServerError, err return http.StatusInternalServerError, err
} }

View File

@ -26,6 +26,10 @@ type info struct {
Content map[string]interface{} Content map[string]interface{}
} }
type response struct {
Message string `json:"message"`
}
// POST handles the POST method on editor page // POST handles the POST method on editor page
func POST(w http.ResponseWriter, r *http.Request) (int, error) { func POST(w http.ResponseWriter, r *http.Request) (int, error) {
// Get the JSON information sent using a buffer // Get the JSON information sent using a buffer
@ -34,9 +38,7 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
err := json.Unmarshal(rawBuffer.Bytes(), &data) err := json.Unmarshal(rawBuffer.Bytes(), &data)
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{"Error decrypting json."}, http.StatusInternalServerError, err)
"message": "Error decrypting json.",
}, http.StatusInternalServerError, err)
} }
// Initializes the file content to write // Initializes the file content to write
@ -46,9 +48,7 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
case "frontmatter-only": case "frontmatter-only":
f, code, err := parseFrontMatterOnlyFile() f, code, err := parseFrontMatterOnlyFile()
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{err.Error()}, code, err)
"message": err.Error(),
}, code, err)
} }
file = f file = f
@ -61,25 +61,19 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
case "complete": case "complete":
f, code, err := parseCompleteFile() f, code, err := parseCompleteFile()
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{err.Error()}, code, err)
"message": err.Error(),
}, code, err)
} }
file = f file = f
default: default:
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{"Invalid content type."}, http.StatusBadRequest, nil)
"message": "Invalid content type.",
}, http.StatusBadRequest, nil)
} }
// Write the file // Write the file
err = ioutil.WriteFile(filename, file, 0666) err = ioutil.WriteFile(filename, file, 0666)
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return server.RespondJSON(w, &response{err.Error()}, http.StatusInternalServerError, err)
"message": err.Error(),
}, http.StatusInternalServerError, err)
} }
if data.Regenerate { if data.Regenerate {

51
routes/errors/errors.go Normal file
View File

@ -0,0 +1,51 @@
package errors
import (
"net/http"
"strconv"
"text/template"
"github.com/hacdias/caddy-hugo/tools/server"
"github.com/hacdias/caddy-hugo/tools/templates"
"github.com/hacdias/caddy-hugo/tools/variables"
)
type errorInformation struct {
Title string `json:"title"`
Message string `json:"message"`
err error
}
// ServeHTTP is used to serve the content of GIT API.
func ServeHTTP(w http.ResponseWriter, r *http.Request, code int, err error) (int, error) {
page := new(errorInformation)
page.Title = strconv.Itoa(code) + " " + http.StatusText(code)
page.err = err
if err != nil {
page.Message = err.Error()
}
switch r.Method {
case "GET":
functions := template.FuncMap{
"Defined": variables.Defined,
}
tpl, err := templates.Get(r, functions, "error")
if err != nil {
return http.StatusInternalServerError, err
}
err = tpl.Execute(w, page)
if err != nil {
return http.StatusInternalServerError, err
}
return 0, page.err
default:
return server.RespondJSON(w, page, code, err)
}
}

View File

@ -7,16 +7,18 @@ import (
"os/exec" "os/exec"
"strings" "strings"
"github.com/hacdias/caddy-hugo/tools/server" s "github.com/hacdias/caddy-hugo/tools/server"
) )
type postError struct {
Message string `json:"message"`
}
// POST handles the POST method on GIT page which is only an API. // POST handles the POST method on GIT page which is only an API.
func POST(w http.ResponseWriter, r *http.Request) (int, error) { func POST(w http.ResponseWriter, r *http.Request) (int, error) {
// Check if git is installed on the computer // Check if git is installed on the computer
if _, err := exec.LookPath("git"); err != nil { if _, err := exec.LookPath("git"); err != nil {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &postError{"Git is not installed on your computer."}, 400, nil)
"message": "Git is not installed on your computer.",
}, 400, nil)
} }
// Get the JSON information sent using a buffer // Get the JSON information sent using a buffer
@ -29,9 +31,7 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
// Check if command was sent // Check if command was sent
if _, ok := info["command"]; !ok { if _, ok := info["command"]; !ok {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &postError{"Command not specified."}, 400, nil)
"message": "Command not specified.",
}, 400, nil)
} }
command := info["command"].(string) command := info["command"].(string)
@ -42,9 +42,7 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
} }
if len(args) == 0 { if len(args) == 0 {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &postError{"Command not specified."}, 400, nil)
"message": "Command not specified.",
}, 400, nil)
} }
cmd := exec.Command("git", args...) cmd := exec.Command("git", args...)
@ -52,12 +50,8 @@ func POST(w http.ResponseWriter, r *http.Request) (int, error) {
output, err := cmd.CombinedOutput() output, err := cmd.CombinedOutput()
if err != nil { if err != nil {
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &postError{err.Error()}, 500, err)
"message": err.Error(),
}, 500, err)
} }
return server.RespondJSON(w, map[string]string{ return s.RespondJSON(w, &postError{string(output)}, 200, nil)
"message": string(output),
}, 200, nil)
} }

View File

@ -2,12 +2,11 @@ package server
import ( import (
"encoding/json" "encoding/json"
"errors"
"net/http" "net/http"
) )
// RespondJSON used to send JSON responses to the web server // RespondJSON used to send JSON responses to the web server
func RespondJSON(w http.ResponseWriter, message map[string]string, code int, err error) (int, error) { func RespondJSON(w http.ResponseWriter, message interface{}, code int, err error) (int, error) {
if message == nil { if message == nil {
message = map[string]string{} message = map[string]string{}
} }
@ -18,10 +17,6 @@ func RespondJSON(w http.ResponseWriter, message map[string]string, code int, err
return 500, msgErr return 500, msgErr
} }
if code == 500 && err != nil {
err = errors.New(message["message"])
}
w.Header().Set("Content-Type", "application/json") w.Header().Set("Content-Type", "application/json")
w.WriteHeader(code) w.WriteHeader(code)
w.Write(msg) w.Write(msg)