gf/g/net/ghttp/ghttp_request.go

217 lines
7.0 KiB
Go
Raw Normal View History

2017-12-29 16:03:30 +08:00
// Copyright 2017 gf Author(https://gitee.com/johng/gf). All Rights Reserved.
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://gitee.com/johng/gf.
2017-12-31 18:19:58 +08:00
2017-11-23 10:21:28 +08:00
package ghttp
import (
"io/ioutil"
2017-12-31 12:15:04 +08:00
"net/http"
"gitee.com/johng/gf/g/encoding/gjson"
"gitee.com/johng/gf/g/container/gtype"
2018-07-11 17:06:47 +08:00
"gitee.com/johng/gf/g/util/gregex"
"gitee.com/johng/gf/g/os/gtime"
"github.com/fatih/structs"
"strings"
2017-11-23 10:21:28 +08:00
)
2017-12-31 12:15:04 +08:00
// 请求对象
type Request struct {
2017-12-31 12:15:04 +08:00
http.Request
parsedGet *gtype.Bool // GET参数是否已经解析
parsedPost *gtype.Bool // POST参数是否已经解析
queryVars map[string][]string // GET参数
routerVars map[string][]string // 路由解析参数
exit *gtype.Bool // 是否退出当前请求流程执行
Id int // 请求id(唯一)
Server *Server // 请求关联的服务器对象
Cookie *Cookie // 与当前请求绑定的Cookie对象(并发安全)
Session *Session // 与当前请求绑定的Session对象(并发安全)
Response *Response // 对应请求的返回数据操作对象
Router *Router // 匹配到的路由对象
EnterTime int64 // 请求进入时间(微秒)
LeaveTime int64 // 请求完成时间(微秒)
Param interface{} // 开发者自定义参数
parsedHost *gtype.String // 解析过后不带端口号的服务器域名名称
clientIp *gtype.String // 解析过后的客户端IP地址
isFileRequest bool // 是否为静态文件请求(非服务请求,当静态文件存在时,优先级会被服务请求高,被识别为文件请求)
}
2018-04-11 16:06:45 +08:00
// 创建一个Request对象
func newRequest(s *Server, r *http.Request, w http.ResponseWriter) *Request {
request := &Request{
2018-04-11 16:06:45 +08:00
parsedGet : gtype.NewBool(),
parsedPost : gtype.NewBool(),
2018-07-29 22:01:29 +08:00
queryVars : make(map[string][]string),
routerVars : make(map[string][]string),
2018-04-16 16:50:24 +08:00
exit : gtype.NewBool(),
2018-04-11 16:06:45 +08:00
Id : s.servedCount.Add(1),
Server : s,
Request : *r,
Response : newResponse(s, w),
EnterTime : gtime.Microsecond(),
2018-05-28 17:02:53 +08:00
parsedHost : gtype.NewString(),
clientIp : gtype.NewString(),
2018-04-11 16:06:45 +08:00
}
// 会话处理
request.Cookie = GetCookie(request)
request.Session = GetSession(request)
request.Response.request = request
return request
2018-04-11 16:06:45 +08:00
}
// 获取Web Socket连接对象(如果是非WS请求会失败注意检查然会的error结果)
func (r *Request) WebSocket() (*WebSocket, error) {
if conn, err := wsUpgrader.Upgrade(r.Response.ResponseWriter.ResponseWriter, &r.Request, nil); err == nil {
return &WebSocket {
conn,
}, nil
} else {
return nil, err
}
}
2018-07-29 22:01:29 +08:00
// 获得指定名称的参数字符串(Router/GET/POST),同 GetRequestString
2018-04-14 12:34:11 +08:00
// 这是常用方法的简化别名
2018-08-19 22:45:23 +08:00
func (r *Request) Get(key string, def ... string) string {
return r.GetRequestString(key, def...)
2018-04-14 12:34:11 +08:00
}
// 获取原始请求输入字符串,注意:只能获取一次,读完就没了
func (r *Request) GetRaw() []byte {
result, _ := ioutil.ReadAll(r.Body)
return result
2017-11-23 10:21:28 +08:00
}
2017-12-31 11:09:16 +08:00
// 获取原始json请求输入字符串并解析为json对象
func (r *Request) GetJson() *gjson.Json {
2017-11-23 10:21:28 +08:00
data := r.GetRaw()
if data != nil {
if j, err := gjson.DecodeToJson(data); err == nil {
return j
}
2017-11-23 10:21:28 +08:00
}
return nil
}
2018-08-19 22:45:23 +08:00
func (r *Request) GetString(key string, def ... string) string {
return r.GetRequestString(key, def...)
2018-08-17 18:21:10 +08:00
}
2018-08-19 22:45:23 +08:00
func (r *Request) GetInt(key string, def ... int) int {
return r.GetRequestInt(key, def...)
2018-08-17 18:21:10 +08:00
}
func (r *Request) GetInts(key string, def ... []int) []int {
return r.GetRequestInts(key, def...)
}
2018-08-19 22:45:23 +08:00
func (r *Request) GetUint(key string, def ... uint) uint {
return r.GetRequestUint(key, def...)
2018-08-17 18:21:10 +08:00
}
2018-08-19 22:45:23 +08:00
func (r *Request) GetFloat32(key string, def ... float32) float32 {
return r.GetRequestFloat32(key, def...)
2018-08-17 18:21:10 +08:00
}
2018-08-19 22:45:23 +08:00
func (r *Request) GetFloat64(key string, def ... float64) float64 {
return r.GetRequestFloat64(key, def...)
2018-08-17 18:21:10 +08:00
}
func (r *Request) GetFloats(key string, def ... []float64) []float64 {
return r.GetRequestFloats(key, def...)
}
2018-08-19 22:45:23 +08:00
func (r *Request) GetArray(key string, def ... []string) []string {
return r.GetRequestArray(key, def...)
2018-08-17 18:21:10 +08:00
}
func (r *Request) GetStrings(key string, def ... []string) []string {
return r.GetRequestStrings(key, def...)
}
func (r *Request) GetInterfaces(key string, def ... []interface{}) []interface{} {
return r.GetRequestInterfaces(key, def...)
}
2018-08-19 22:45:23 +08:00
func (r *Request) GetMap(def...map[string]string) map[string]string {
return r.GetRequestMap(def...)
2018-08-17 18:21:10 +08:00
}
// 将所有的request参数映射到struct属性上参数object应当为一个struct对象的指针, mapping为非必需参数自定义参数与属性的映射关系
func (r *Request) GetToStruct(object interface{}, mapping...map[string]string) {
r.GetRequestToStruct(object, mapping...)
}
2018-04-16 16:50:24 +08:00
// 退出当前请求执行原理是在Request.exit做标记由服务逻辑流程做判断自行停止
func (r *Request) Exit() {
r.exit.Set(true)
}
// 判断当前请求是否停止执行
2018-04-16 16:50:26 +08:00
func (r *Request) IsExited() bool {
2018-04-16 16:50:24 +08:00
return r.exit.Val()
}
2018-04-25 17:11:34 +08:00
// 获取请求的服务端IP/域名
func (r *Request) GetHost() string {
2018-05-28 17:02:53 +08:00
host := r.parsedHost.Val()
if len(host) == 0 {
2018-07-11 17:06:47 +08:00
array, _ := gregex.MatchString(`(.+):(\d+)`, r.Host)
2018-05-28 17:02:53 +08:00
if len(array) > 1 {
host = array[1]
} else {
host = r.Host
}
r.parsedHost.Set(host)
2018-04-25 17:11:34 +08:00
}
2018-05-28 17:02:53 +08:00
return host
2018-04-25 17:11:34 +08:00
}
// 判断是否为静态文件请求
func (r *Request) IsFileRequest() bool {
return r.isFileRequest
}
2018-08-22 19:56:01 +08:00
// 判断是否为AJAX请求
func (r *Request) IsAjaxRequest() bool {
2018-08-22 19:56:01 +08:00
return strings.EqualFold(r.Header.Get("X-Requested-With"), "XMLHttpRequest")
}
2018-04-25 17:11:34 +08:00
// 获取请求的客户端IP地址
2018-04-20 15:43:02 +08:00
func (r *Request) GetClientIp() string {
2018-05-28 17:02:53 +08:00
ip := r.clientIp.Val()
if len(ip) == 0 {
if ip = r.Header.Get("X-Real-IP"); ip == "" {
array, _ := gregex.MatchString(`(.+):(\d+)`, r.RemoteAddr)
if len(array) > 1 {
ip = array[1]
} else {
ip = r.RemoteAddr
}
2018-05-28 17:02:53 +08:00
}
r.clientIp.Set(ip)
2018-04-20 15:43:02 +08:00
}
2018-05-28 17:02:53 +08:00
return ip
2018-04-20 15:43:02 +08:00
}
2017-11-23 10:21:28 +08:00
// 获得来源URL地址
func (r *Request) GetReferer() string {
return r.Header.Get("Referer")
}
// 获得结构体顶替的参数名称标签构成map返回
func (r *Request) getStructParamsTagMap(object interface{}) map[string]string {
tagmap := make(map[string]string)
fields := structs.Fields(object)
for _, field := range fields {
if tag := field.Tag("params"); tag != "" {
for _, v := range strings.Split(tag, ",") {
tagmap[strings.TrimSpace(v)] = field.Name()
}
}
}
return tagmap
}