mirror of
				https://github.com/georgyo/ifconfig.io.git
				synced 2025-10-31 02:36:38 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			151 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			151 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package main
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"net"
 | |
| 	"net/http"
 | |
| 	"net/http/fcgi"
 | |
| 	"path"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/gin-gonic/gin"
 | |
| )
 | |
| 
 | |
| // Logger is a simple log handler, out puts in the standard of apache access log common.
 | |
| // See http://httpd.apache.org/docs/2.2/logs.html#accesslog
 | |
| func Logger() gin.HandlerFunc {
 | |
| 	return func(c *gin.Context) {
 | |
| 		t := time.Now()
 | |
| 		ip, err := net.ResolveTCPAddr("tcp", c.Req.RemoteAddr)
 | |
| 		if err != nil {
 | |
| 			c.Abort(500)
 | |
| 		}
 | |
| 
 | |
| 		// before request
 | |
| 		c.Next()
 | |
| 		// after request
 | |
| 
 | |
| 		user := "-"
 | |
| 		if c.Req.URL.User != nil {
 | |
| 			user = c.Req.URL.User.Username()
 | |
| 		}
 | |
| 
 | |
| 		latency := time.Since(t)
 | |
| 
 | |
| 		// This is the format of Apache Log Common, with an additional field of latency
 | |
| 		fmt.Printf("%v - %v [%v] \"%v %v %v\" %v %v %v\n",
 | |
| 			ip.IP, user, t.Format(time.RFC3339), c.Req.Method, c.Req.URL.Path,
 | |
| 			c.Req.Proto, c.Writer.Status(), c.Req.ContentLength, latency)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func stringInSlice(a string, list []string) bool {
 | |
| 	for _, b := range list {
 | |
| 		if b == a {
 | |
| 			return true
 | |
| 		}
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func mainHandler(c *gin.Context) {
 | |
| 	fields := strings.Split(c.Params.ByName("field"), ".")
 | |
| 	ip, err := net.ResolveTCPAddr("tcp", c.Req.RemoteAddr)
 | |
| 	if err != nil {
 | |
| 		c.Abort(500)
 | |
| 	}
 | |
| 	c.Set("ip", ip.IP.String())
 | |
| 	c.Set("port", ip.Port)
 | |
| 	c.Set("ua", c.Req.UserAgent())
 | |
| 	c.Set("lang", c.Req.Header.Get("Accept-Language"))
 | |
| 	c.Set("encoding", c.Req.Header.Get("Accept-Encoding"))
 | |
| 	c.Set("method", c.Req.Method)
 | |
| 	c.Set("mime", c.Req.Header.Get("Accept"))
 | |
| 	c.Set("referer", c.Req.Header.Get("Referer"))
 | |
| 	c.Set("forwarded", c.Req.Header.Get("X-Forwarded-For"))
 | |
| 
 | |
| 	// Only lookup hostname if the results are going to need it.
 | |
| 	if stringInSlice(fields[0], []string{"", "all", "host"}) {
 | |
| 		hostnames, err := net.LookupAddr(ip.IP.String())
 | |
| 		if err != nil {
 | |
| 			c.Set("host", "")
 | |
| 		} else {
 | |
| 			c.Set("host", hostnames[0])
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	wantsJSON := false
 | |
| 	if len(fields) >= 2 && fields[1] == "json" {
 | |
| 		wantsJSON = true
 | |
| 	}
 | |
| 
 | |
| 	ua := strings.Split(c.Req.UserAgent(), "/")
 | |
| 	switch fields[0] {
 | |
| 	case "":
 | |
| 		//If the user is using curl, then we should just return the IP, else we show the home page.
 | |
| 		if ua[0] == "curl" {
 | |
| 			c.String(200, fmt.Sprintln(ip.IP))
 | |
| 		} else {
 | |
| 			c.HTML(200, "index.html", c.Keys)
 | |
| 		}
 | |
| 		return
 | |
| 	case "request":
 | |
| 		c.JSON(200, c.Req)
 | |
| 		return
 | |
| 	case "all":
 | |
| 		if wantsJSON {
 | |
| 			c.JSON(200, c.Keys)
 | |
| 		} else {
 | |
| 			c.String(200, "%v", c.Keys)
 | |
| 		}
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	fieldResult, err := c.Get(fields[0])
 | |
| 	if err != nil {
 | |
| 		c.String(404, "Not Found")
 | |
| 	}
 | |
| 	c.String(200, fmt.Sprintln(fieldResult))
 | |
| 
 | |
| }
 | |
| 
 | |
| // FileServer is a basic file serve handler, this is just here as an example.
 | |
| // gin.Static() should be used instead
 | |
| func FileServer(root string) gin.HandlerFunc {
 | |
| 	return func(c *gin.Context) {
 | |
| 		file := c.Params.ByName("file")
 | |
| 		if !strings.HasPrefix(file, "/") {
 | |
| 			file = "/" + file
 | |
| 		}
 | |
| 		http.ServeFile(c.Writer, c.Req, path.Join(root, path.Clean(file)))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func main() {
 | |
| 	r := gin.New()
 | |
| 	r.Use(gin.Recovery())
 | |
| 	r.Use(Logger())
 | |
| 	r.LoadHTMLTemplates("templates/*")
 | |
| 
 | |
| 	r.GET("/:field", mainHandler)
 | |
| 	r.GET("/", mainHandler)
 | |
| 
 | |
| 	// Create a listener for FCGI
 | |
| 	fcgi_listen, err := net.Listen("tcp", "127.0.0.1:4000")
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 	err = fcgi.Serve(fcgi_listen, r)
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 
 | |
| 	//port := os.Getenv("PORT")
 | |
| 	//host := os.Getenv("HOST")
 | |
| 	//if port == "" {
 | |
| 	//	port = "8080"
 | |
| 	//}
 | |
| 	//r.Run(host + ":" + port)
 | |
| }
 |