ghttp.Server 日志功能完善

This commit is contained in:
John
2018-04-20 15:43:02 +08:00
parent 05741aadda
commit 4e9e1d619f
6 changed files with 52 additions and 32 deletions

View File

@ -12,6 +12,7 @@ import (
"gitee.com/johng/gf/g/util/gconv"
"gitee.com/johng/gf/g/encoding/gjson"
"gitee.com/johng/gf/g/container/gtype"
"gitee.com/johng/gf/g/util/gregx"
)
// 请求对象
@ -39,8 +40,7 @@ func newRequest(s *Server, r *http.Request, w http.ResponseWriter) *Request {
Server : s,
Request : *r,
Response : &Response {
status : http.StatusOK,
ResponseWriter : w,
ResponseWriter : ResponseWriter{w, http.StatusOK, 0},
},
}
// 会话处理
@ -298,5 +298,12 @@ func (r *Request) IsExited() bool {
return r.exit.Val()
}
// 获取请求的客户端Ip地址
func (r *Request) GetClientIp() string {
array, _ := gregx.MatchString(`(.+):(\d+)`, r.RemoteAddr)
if len(array) > 1 {
return array[1]
}
return r.RemoteAddr
}

View File

@ -17,11 +17,30 @@ import (
// 服务端请求返回对象
type Response struct {
http.ResponseWriter
bufmu sync.RWMutex // 缓冲区互斥锁
ResponseWriter
mu sync.RWMutex // 缓冲区互斥锁
buffer []byte // 每个请求的返回数据缓冲区
request *Request // 关联的Request请求对象
status int // 返回状态码
}
// 自定义的ResponseWriter用于写入流的控制
type ResponseWriter struct {
http.ResponseWriter
status int // http status
length int // response length
}
// 覆盖父级的WriteHeader方法
func (w *ResponseWriter) Write(buffer []byte) (int, error) {
n, e := w.ResponseWriter.Write(buffer)
w.length += n
return n, e
}
// 覆盖父级的WriteHeader方法
func (w *ResponseWriter) WriteHeader(code int) {
w.status = code
w.ResponseWriter.WriteHeader(code)
}
// 返回信息任何变量自动转换为bytes
@ -29,11 +48,11 @@ func (r *Response) Write(content ... interface{}) {
if len(content) == 0 {
return
}
r.bufmu.Lock()
r.mu.Lock()
for _, v := range content {
r.buffer = append(r.buffer, gconv.Bytes(v)...)
}
r.bufmu.Unlock()
r.mu.Unlock()
}
// 返回信息,末尾增加换行标识符"\n"
@ -99,7 +118,6 @@ func (r *Response) SetAllowCrossDomainRequest(allowOrigin string, allowMethods s
// 返回HTTP Code状态码
func (r *Response) WriteStatus(code int, content...string) {
r.status = code
r.Header().Set("Content-Type", "text/plain; charset=utf-8")
r.Header().Set("X-Content-Type-Options", "nosniff")
if len(content) > 0 {
@ -118,31 +136,31 @@ func (r *Response) RedirectTo(location string) {
// 获取当前缓冲区中的数据
func (r *Response) Buffer() []byte {
r.bufmu.RLock()
defer r.bufmu.RUnlock()
r.mu.RLock()
defer r.mu.RUnlock()
return r.buffer
}
// 手动设置缓冲区内容
func (r *Response) SetBuffer(buffer []byte) {
r.bufmu.Lock()
r.mu.Lock()
r.buffer = buffer
r.bufmu.Unlock()
r.mu.Unlock()
}
// 清空缓冲区内容
func (r *Response) ClearBuffer() {
r.bufmu.Lock()
r.mu.Lock()
r.buffer = make([]byte, 0)
r.bufmu.Unlock()
r.mu.Unlock()
}
// 输出缓冲区数据到客户端
func (r *Response) OutputBuffer() {
r.bufmu.Lock()
r.mu.Lock()
if len(r.buffer) > 0 {
r.ResponseWriter.Write(r.buffer)
r.buffer = make([]byte, 0)
}
r.bufmu.Unlock()
r.mu.Unlock()
}

View File

@ -130,7 +130,7 @@ func (c *Cookie) Output() {
continue
}
http.SetCookie(
c.response.ResponseWriter,
c.response.ResponseWriter.ResponseWriter,
&http.Cookie {
Name : k,
Value : v.value,

View File

@ -136,7 +136,7 @@ func (s *Server)doServeFile(r *Request, path string) {
}
} else {
// 读取文件内容返回, no buffer
http.ServeContent(r.Response.ResponseWriter, &r.Request, info.Name(), info.ModTime(), f)
http.ServeContent(r.Response.ResponseWriter.ResponseWriter, &r.Request, info.Name(), info.ModTime(), f)
}
f.Close()
}

View File

@ -9,7 +9,6 @@ package ghttp
import (
"fmt"
"strings"
"gitee.com/johng/gf/g/util/gconv"
)
@ -23,12 +22,8 @@ func (s *Server) handleAccessLog(r *Request) {
v(r)
return
}
status := gconv.String(r.Response.status)
if v := r.Response.Header().Get("Status Code"); v != "" {
status = v
}
content := fmt.Sprintf(`"%s %s %s %s" %s`, r.Method, r.Host, r.URL.String(), r.Proto, status)
content += fmt.Sprintf(`, %s, "%s", "%s"`, strings.Split(r.RemoteAddr, ":")[0], r.Referer(), r.UserAgent())
content := fmt.Sprintf(`"%s %s %s %s" %s %s`, r.Method, r.Host, r.URL.String(), r.Proto, gconv.String(r.Response.status), gconv.String(r.Response.length))
content += fmt.Sprintf(`, %s, "%s", "%s"`, r.GetClientIp(), r.Referer(), r.UserAgent())
s.logger.Println(content)
}
@ -44,7 +39,7 @@ func (s *Server) handleErrorLog(error interface{}, r *Request) {
}
content := fmt.Sprintf(`"%s %s %s %s"`, r.Method, r.Host, r.URL.String(), r.Proto)
content += fmt.Sprintf(`, %s, "%s", "%s"`, strings.Split(r.RemoteAddr, ":")[0], r.Referer(), r.UserAgent())
content += fmt.Sprintf(`, %s, "%s", "%s"`, r.GetClientIp(), r.Referer(), r.UserAgent())
content += fmt.Sprintf(`, %v`, error)
s.logger.Error(content)
}

View File

@ -46,26 +46,26 @@ func (p *utilpprof) Index(r *Request) {
}
for _, p := range profiles {
if p.Name() == action {
p.WriteTo(r.Response.ResponseWriter, r.GetRequestInt("debug"))
p.WriteTo(r.Response.ResponseWriter.ResponseWriter, r.GetRequestInt("debug"))
break
}
}
}
func (p *utilpprof) Cmdline(r *Request) {
netpprof.Cmdline(r.Response.ResponseWriter, &r.Request)
netpprof.Cmdline(r.Response.ResponseWriter.ResponseWriter, &r.Request)
}
func (p *utilpprof) Profile(r *Request) {
netpprof.Profile(r.Response.ResponseWriter, &r.Request)
netpprof.Profile(r.Response.ResponseWriter.ResponseWriter, &r.Request)
}
func (p *utilpprof) Symbol(r *Request) {
netpprof.Symbol(r.Response.ResponseWriter, &r.Request)
netpprof.Symbol(r.Response.ResponseWriter.ResponseWriter, &r.Request)
}
func (p *utilpprof) Trace(r *Request) {
netpprof.Trace(r.Response.ResponseWriter, &r.Request)
netpprof.Trace(r.Response.ResponseWriter.ResponseWriter, &r.Request)
}
// 开启pprof支持