123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 |
- // Copyright 2014 Manu Martinez-Almeida. All rights reserved.
- // Use of this source code is governed by a MIT style
- // license that can be found in the LICENSE file.
-
- package gin
-
- import (
- "encoding/xml"
- "net/http"
- "os"
- "path"
- "reflect"
- "runtime"
- "strings"
- )
-
- const BindKey = "_gin-gonic/gin/bindkey"
-
- func Bind(val interface{}) HandlerFunc {
- value := reflect.ValueOf(val)
- if value.Kind() == reflect.Ptr {
- panic(`Bind struct can not be a pointer. Example:
- Use: gin.Bind(Struct{}) instead of gin.Bind(&Struct{})
- `)
- }
- typ := value.Type()
-
- return func(c *Context) {
- obj := reflect.New(typ).Interface()
- if c.Bind(obj) == nil {
- c.Set(BindKey, obj)
- }
- }
- }
-
- // WrapF is a helper function for wrapping http.HandlerFunc
- // Returns a Gin middleware
- func WrapF(f http.HandlerFunc) HandlerFunc {
- return func(c *Context) {
- f(c.Writer, c.Request)
- }
- }
-
- // WrapH is a helper function for wrapping http.Handler
- // Returns a Gin middleware
- func WrapH(h http.Handler) HandlerFunc {
- return func(c *Context) {
- h.ServeHTTP(c.Writer, c.Request)
- }
- }
-
- // H is a shortcut for map[string]interface{}
- type H map[string]interface{}
-
- // MarshalXML allows type H to be used with xml.Marshal.
- func (h H) MarshalXML(e *xml.Encoder, start xml.StartElement) error {
- start.Name = xml.Name{
- Space: "",
- Local: "map",
- }
- if err := e.EncodeToken(start); err != nil {
- return err
- }
- for key, value := range h {
- elem := xml.StartElement{
- Name: xml.Name{Space: "", Local: key},
- Attr: []xml.Attr{},
- }
- if err := e.EncodeElement(value, elem); err != nil {
- return err
- }
- }
-
- return e.EncodeToken(xml.EndElement{Name: start.Name})
- }
-
- func assert1(guard bool, text string) {
- if !guard {
- panic(text)
- }
- }
-
- func filterFlags(content string) string {
- for i, char := range content {
- if char == ' ' || char == ';' {
- return content[:i]
- }
- }
- return content
- }
-
- func chooseData(custom, wildcard interface{}) interface{} {
- if custom == nil {
- if wildcard == nil {
- panic("negotiation config is invalid")
- }
- return wildcard
- }
- return custom
- }
-
- func parseAccept(acceptHeader string) []string {
- parts := strings.Split(acceptHeader, ",")
- out := make([]string, 0, len(parts))
- for _, part := range parts {
- if part = strings.TrimSpace(strings.Split(part, ";")[0]); part != "" {
- out = append(out, part)
- }
- }
- return out
- }
-
- func lastChar(str string) uint8 {
- if str == "" {
- panic("The length of the string can't be 0")
- }
- return str[len(str)-1]
- }
-
- func nameOfFunction(f interface{}) string {
- return runtime.FuncForPC(reflect.ValueOf(f).Pointer()).Name()
- }
-
- func joinPaths(absolutePath, relativePath string) string {
- if relativePath == "" {
- return absolutePath
- }
-
- finalPath := path.Join(absolutePath, relativePath)
- appendSlash := lastChar(relativePath) == '/' && lastChar(finalPath) != '/'
- if appendSlash {
- return finalPath + "/"
- }
- return finalPath
- }
-
- func resolveAddress(addr []string) string {
- switch len(addr) {
- case 0:
- if port := os.Getenv("PORT"); port != "" {
- debugPrint("Environment variable PORT=\"%s\"", port)
- return ":" + port
- }
- debugPrint("Environment variable PORT is undefined. Using port :8080 by default")
- return ":8080"
- case 1:
- return addr[0]
- default:
- panic("too much parameters")
- }
- }
|