2013-08-21 17:59:31 +08:00
|
|
|
|
package context
|
2013-08-21 13:24:14 +08:00
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"bytes"
|
|
|
|
|
"compress/flate"
|
|
|
|
|
"compress/gzip"
|
|
|
|
|
"encoding/json"
|
|
|
|
|
"encoding/xml"
|
|
|
|
|
"errors"
|
|
|
|
|
"fmt"
|
2013-11-08 18:00:07 +08:00
|
|
|
|
"html/template"
|
2013-08-21 13:24:14 +08:00
|
|
|
|
"io"
|
|
|
|
|
"mime"
|
|
|
|
|
"net/http"
|
|
|
|
|
"path/filepath"
|
|
|
|
|
"strconv"
|
|
|
|
|
"strings"
|
|
|
|
|
)
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// BeegoOutput does work for sending response header.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
type BeegoOutput struct {
|
2013-09-10 00:00:11 +08:00
|
|
|
|
Context *Context
|
2013-08-21 13:24:14 +08:00
|
|
|
|
Status int
|
|
|
|
|
EnableGzip bool
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// NewOutput returns new BeegoOutput.
|
|
|
|
|
// it contains nothing now.
|
2013-12-18 22:33:21 +08:00
|
|
|
|
func NewOutput() *BeegoOutput {
|
|
|
|
|
return &BeegoOutput{}
|
2013-08-21 13:24:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Header sets response header item string via given key.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) Header(key, val string) {
|
2013-12-18 22:33:21 +08:00
|
|
|
|
output.Context.ResponseWriter.Header().Set(key, val)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Body sets response body content.
|
|
|
|
|
// if EnableGzip, compress content string.
|
|
|
|
|
// it sends out response body directly.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) Body(content []byte) {
|
2013-12-18 22:33:21 +08:00
|
|
|
|
output_writer := output.Context.ResponseWriter.(io.Writer)
|
2013-09-10 00:00:11 +08:00
|
|
|
|
if output.EnableGzip == true && output.Context.Input.Header("Accept-Encoding") != "" {
|
|
|
|
|
splitted := strings.SplitN(output.Context.Input.Header("Accept-Encoding"), ",", -1)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
encodings := make([]string, len(splitted))
|
|
|
|
|
|
|
|
|
|
for i, val := range splitted {
|
|
|
|
|
encodings[i] = strings.TrimSpace(val)
|
|
|
|
|
}
|
|
|
|
|
for _, val := range encodings {
|
|
|
|
|
if val == "gzip" {
|
|
|
|
|
output.Header("Content-Encoding", "gzip")
|
2013-12-18 22:33:21 +08:00
|
|
|
|
output_writer, _ = gzip.NewWriterLevel(output.Context.ResponseWriter, gzip.BestSpeed)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
|
|
|
|
|
break
|
|
|
|
|
} else if val == "deflate" {
|
|
|
|
|
output.Header("Content-Encoding", "deflate")
|
2013-12-18 22:33:21 +08:00
|
|
|
|
output_writer, _ = flate.NewWriter(output.Context.ResponseWriter, flate.BestSpeed)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
output.Header("Content-Length", strconv.Itoa(len(content)))
|
|
|
|
|
}
|
|
|
|
|
output_writer.Write(content)
|
|
|
|
|
switch output_writer.(type) {
|
|
|
|
|
case *gzip.Writer:
|
|
|
|
|
output_writer.(*gzip.Writer).Close()
|
|
|
|
|
case *flate.Writer:
|
|
|
|
|
output_writer.(*flate.Writer).Close()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Cookie sets cookie value via given key.
|
|
|
|
|
// others are ordered as cookie's max age time, path,domain, secure and httponly.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) Cookie(name string, value string, others ...interface{}) {
|
|
|
|
|
var b bytes.Buffer
|
|
|
|
|
fmt.Fprintf(&b, "%s=%s", sanitizeName(name), sanitizeValue(value))
|
|
|
|
|
if len(others) > 0 {
|
|
|
|
|
switch others[0].(type) {
|
|
|
|
|
case int:
|
|
|
|
|
if others[0].(int) > 0 {
|
|
|
|
|
fmt.Fprintf(&b, "; Max-Age=%d", others[0].(int))
|
|
|
|
|
} else if others[0].(int) < 0 {
|
|
|
|
|
fmt.Fprintf(&b, "; Max-Age=0")
|
|
|
|
|
}
|
|
|
|
|
case int64:
|
|
|
|
|
if others[0].(int64) > 0 {
|
|
|
|
|
fmt.Fprintf(&b, "; Max-Age=%d", others[0].(int64))
|
|
|
|
|
} else if others[0].(int64) < 0 {
|
|
|
|
|
fmt.Fprintf(&b, "; Max-Age=0")
|
|
|
|
|
}
|
|
|
|
|
case int32:
|
|
|
|
|
if others[0].(int32) > 0 {
|
|
|
|
|
fmt.Fprintf(&b, "; Max-Age=%d", others[0].(int32))
|
|
|
|
|
} else if others[0].(int32) < 0 {
|
|
|
|
|
fmt.Fprintf(&b, "; Max-Age=0")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if len(others) > 1 {
|
|
|
|
|
fmt.Fprintf(&b, "; Path=%s", sanitizeValue(others[1].(string)))
|
|
|
|
|
}
|
|
|
|
|
if len(others) > 2 {
|
|
|
|
|
fmt.Fprintf(&b, "; Domain=%s", sanitizeValue(others[2].(string)))
|
|
|
|
|
}
|
|
|
|
|
if len(others) > 3 {
|
|
|
|
|
fmt.Fprintf(&b, "; Secure")
|
|
|
|
|
}
|
|
|
|
|
if len(others) > 4 {
|
|
|
|
|
fmt.Fprintf(&b, "; HttpOnly")
|
|
|
|
|
}
|
2013-12-18 22:33:21 +08:00
|
|
|
|
output.Context.ResponseWriter.Header().Add("Set-Cookie", b.String())
|
2013-08-21 13:24:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var cookieNameSanitizer = strings.NewReplacer("\n", "-", "\r", "-")
|
|
|
|
|
|
|
|
|
|
func sanitizeName(n string) string {
|
|
|
|
|
return cookieNameSanitizer.Replace(n)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var cookieValueSanitizer = strings.NewReplacer("\n", " ", "\r", " ", ";", " ")
|
|
|
|
|
|
|
|
|
|
func sanitizeValue(v string) string {
|
|
|
|
|
return cookieValueSanitizer.Replace(v)
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Json writes json to response body.
|
|
|
|
|
// if coding is true, it converts utf-8 to \u0000 type.
|
2013-09-10 00:00:11 +08:00
|
|
|
|
func (output *BeegoOutput) Json(data interface{}, hasIndent bool, coding bool) error {
|
2013-08-21 13:24:14 +08:00
|
|
|
|
output.Header("Content-Type", "application/json;charset=UTF-8")
|
2013-09-10 00:00:11 +08:00
|
|
|
|
var content []byte
|
|
|
|
|
var err error
|
|
|
|
|
if hasIndent {
|
|
|
|
|
content, err = json.MarshalIndent(data, "", " ")
|
|
|
|
|
} else {
|
|
|
|
|
content, err = json.Marshal(data)
|
|
|
|
|
}
|
2013-08-21 13:24:14 +08:00
|
|
|
|
if err != nil {
|
2013-12-18 22:33:21 +08:00
|
|
|
|
http.Error(output.Context.ResponseWriter, err.Error(), http.StatusInternalServerError)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
return err
|
|
|
|
|
}
|
2013-09-10 00:00:11 +08:00
|
|
|
|
if coding {
|
|
|
|
|
content = []byte(stringsToJson(string(content)))
|
|
|
|
|
}
|
2013-08-21 13:24:14 +08:00
|
|
|
|
output.Body(content)
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Jsonp writes jsonp to response body.
|
2013-09-10 00:00:11 +08:00
|
|
|
|
func (output *BeegoOutput) Jsonp(data interface{}, hasIndent bool) error {
|
2013-08-21 13:24:14 +08:00
|
|
|
|
output.Header("Content-Type", "application/javascript;charset=UTF-8")
|
2013-09-10 00:00:11 +08:00
|
|
|
|
var content []byte
|
|
|
|
|
var err error
|
|
|
|
|
if hasIndent {
|
|
|
|
|
content, err = json.MarshalIndent(data, "", " ")
|
|
|
|
|
} else {
|
|
|
|
|
content, err = json.Marshal(data)
|
|
|
|
|
}
|
2013-08-21 13:24:14 +08:00
|
|
|
|
if err != nil {
|
2013-12-18 22:33:21 +08:00
|
|
|
|
http.Error(output.Context.ResponseWriter, err.Error(), http.StatusInternalServerError)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
return err
|
|
|
|
|
}
|
2013-09-10 00:00:11 +08:00
|
|
|
|
callback := output.Context.Input.Query("callback")
|
2013-08-21 13:24:14 +08:00
|
|
|
|
if callback == "" {
|
|
|
|
|
return errors.New(`"callback" parameter required`)
|
|
|
|
|
}
|
2013-11-08 20:54:06 +08:00
|
|
|
|
callback_content := bytes.NewBufferString(" " + template.JSEscapeString(callback))
|
2013-08-21 13:24:14 +08:00
|
|
|
|
callback_content.WriteString("(")
|
|
|
|
|
callback_content.Write(content)
|
|
|
|
|
callback_content.WriteString(");\r\n")
|
|
|
|
|
output.Body(callback_content.Bytes())
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Xml writes xml string to response body.
|
2013-09-10 00:02:22 +08:00
|
|
|
|
func (output *BeegoOutput) Xml(data interface{}, hasIndent bool) error {
|
2013-08-21 13:24:14 +08:00
|
|
|
|
output.Header("Content-Type", "application/xml;charset=UTF-8")
|
2013-09-10 00:00:11 +08:00
|
|
|
|
var content []byte
|
|
|
|
|
var err error
|
|
|
|
|
if hasIndent {
|
|
|
|
|
content, err = xml.MarshalIndent(data, "", " ")
|
|
|
|
|
} else {
|
|
|
|
|
content, err = xml.Marshal(data)
|
|
|
|
|
}
|
2013-08-21 13:24:14 +08:00
|
|
|
|
if err != nil {
|
2013-12-18 22:33:21 +08:00
|
|
|
|
http.Error(output.Context.ResponseWriter, err.Error(), http.StatusInternalServerError)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
output.Body(content)
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Download forces response for download file.
|
|
|
|
|
// it prepares the download response header automatically.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) Download(file string) {
|
|
|
|
|
output.Header("Content-Description", "File Transfer")
|
|
|
|
|
output.Header("Content-Type", "application/octet-stream")
|
|
|
|
|
output.Header("Content-Disposition", "attachment; filename="+filepath.Base(file))
|
|
|
|
|
output.Header("Content-Transfer-Encoding", "binary")
|
|
|
|
|
output.Header("Expires", "0")
|
|
|
|
|
output.Header("Cache-Control", "must-revalidate")
|
|
|
|
|
output.Header("Pragma", "public")
|
2013-12-18 22:33:21 +08:00
|
|
|
|
http.ServeFile(output.Context.ResponseWriter, output.Context.Request, file)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// ContentType sets the content type from ext string.
|
|
|
|
|
// MIME type is given in mime package.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) ContentType(ext string) {
|
|
|
|
|
if !strings.HasPrefix(ext, ".") {
|
|
|
|
|
ext = "." + ext
|
|
|
|
|
}
|
|
|
|
|
ctype := mime.TypeByExtension(ext)
|
|
|
|
|
if ctype != "" {
|
|
|
|
|
output.Header("Content-Type", ctype)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// SetStatus sets response status code.
|
|
|
|
|
// It writes response header directly.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) SetStatus(status int) {
|
2013-12-18 22:33:21 +08:00
|
|
|
|
output.Context.ResponseWriter.WriteHeader(status)
|
2013-08-21 13:24:14 +08:00
|
|
|
|
output.Status = status
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsCachable returns boolean of this request is cached.
|
|
|
|
|
// HTTP 304 means cached.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsCachable(status int) bool {
|
|
|
|
|
return output.Status >= 200 && output.Status < 300 || output.Status == 304
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsEmpty returns boolean of this request is empty.
|
|
|
|
|
// HTTP 201,204 and 304 means empty.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsEmpty(status int) bool {
|
|
|
|
|
return output.Status == 201 || output.Status == 204 || output.Status == 304
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsOk returns boolean of this request runs well.
|
|
|
|
|
// HTTP 200 means ok.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsOk(status int) bool {
|
|
|
|
|
return output.Status == 200
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsSuccessful returns boolean of this request runs successfully.
|
|
|
|
|
// HTTP 2xx means ok.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsSuccessful(status int) bool {
|
|
|
|
|
return output.Status >= 200 && output.Status < 300
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsRedirect returns boolean of this request is redirection header.
|
|
|
|
|
// HTTP 301,302,307 means redirection.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsRedirect(status int) bool {
|
|
|
|
|
return output.Status == 301 || output.Status == 302 || output.Status == 303 || output.Status == 307
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsForbidden returns boolean of this request is forbidden.
|
|
|
|
|
// HTTP 403 means forbidden.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsForbidden(status int) bool {
|
|
|
|
|
return output.Status == 403
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsNotFound returns boolean of this request is not found.
|
|
|
|
|
// HTTP 404 means forbidden.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsNotFound(status int) bool {
|
|
|
|
|
return output.Status == 404
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsClient returns boolean of this request client sends error data.
|
|
|
|
|
// HTTP 4xx means forbidden.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsClientError(status int) bool {
|
|
|
|
|
return output.Status >= 400 && output.Status < 500
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// IsServerError returns boolean of this server handler errors.
|
|
|
|
|
// HTTP 5xx means server internal error.
|
2013-08-21 13:24:14 +08:00
|
|
|
|
func (output *BeegoOutput) IsServerError(status int) bool {
|
|
|
|
|
return output.Status >= 500 && output.Status < 600
|
|
|
|
|
}
|
2013-09-10 00:00:11 +08:00
|
|
|
|
|
|
|
|
|
func stringsToJson(str string) string {
|
|
|
|
|
rs := []rune(str)
|
|
|
|
|
jsons := ""
|
|
|
|
|
for _, r := range rs {
|
|
|
|
|
rint := int(r)
|
|
|
|
|
if rint < 128 {
|
|
|
|
|
jsons += string(r)
|
|
|
|
|
} else {
|
|
|
|
|
jsons += "\\u" + strconv.FormatInt(int64(rint), 16) // json
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return jsons
|
|
|
|
|
}
|
2013-11-13 21:11:03 +08:00
|
|
|
|
|
2013-12-25 20:13:38 +08:00
|
|
|
|
// Sessions sets session item value with given key.
|
2013-11-13 21:11:03 +08:00
|
|
|
|
func (output *BeegoOutput) Session(name interface{}, value interface{}) {
|
|
|
|
|
output.Context.Input.CruSession.Set(name, value)
|
|
|
|
|
}
|