mirror of
https://github.com/dcarrillo/whatismyip.git
synced 2024-12-22 07:28:00 +00:00
Use Accept request header instead of user agent to figure out non-browser clients. Funny fact, I borrowed the idea from a fork ('8a3e142cf3/router/generic.go (L33)
')
This commit is contained in:
parent
12da27ddab
commit
9070e9a2c2
@ -47,7 +47,7 @@ curl -6 ifconfig.es
|
|||||||
## Endpoints
|
## Endpoints
|
||||||
|
|
||||||
- https://ifconfig.es/
|
- https://ifconfig.es/
|
||||||
- https://ifconfig.es/json
|
- https://ifconfig.es/json (this is the same as `curl -H "Accept: application/json" https://ifconfig.es/`)
|
||||||
- https://ifconfig.es/geo
|
- https://ifconfig.es/geo
|
||||||
- https://ifconfig.es/geo/city
|
- https://ifconfig.es/geo/city
|
||||||
- https://ifconfig.es/geo/country
|
- https://ifconfig.es/geo/country
|
||||||
|
@ -70,16 +70,18 @@ func TestContainerIntegration(t *testing.T) {
|
|||||||
}()
|
}()
|
||||||
|
|
||||||
http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
|
http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
|
||||||
for _, url := range []string{"http://localhost:8000/json", "https://localhost:8001/json"} {
|
for _, url := range []string{"http://localhost:8000", "https://localhost:8001"} {
|
||||||
resp, _ := http.Get(url)
|
client := &http.Client{}
|
||||||
|
req, _ := http.NewRequest("GET", url, nil)
|
||||||
|
req.Header.Set("Accept", "application/json")
|
||||||
|
resp, _ := client.Do(req)
|
||||||
assert.Equal(t, 200, resp.StatusCode)
|
assert.Equal(t, 200, resp.StatusCode)
|
||||||
|
|
||||||
var dat router.JSONResponse
|
|
||||||
body, err := ioutil.ReadAll(resp.Body)
|
body, err := ioutil.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
assert.NoError(t, json.Unmarshal(body, &dat))
|
assert.NoError(t, json.Unmarshal(body, &router.JSONResponse{}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,6 @@ import (
|
|||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
|
||||||
|
|
||||||
"github.com/dcarrillo/whatismyip/internal/httputils"
|
"github.com/dcarrillo/whatismyip/internal/httputils"
|
||||||
"github.com/dcarrillo/whatismyip/internal/setting"
|
"github.com/dcarrillo/whatismyip/internal/setting"
|
||||||
@ -12,8 +11,6 @@ import (
|
|||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
|
||||||
const userAgentPattern = `curl|wget|libwww-perl|python|ansible-httpget|HTTPie|WindowsPowerShell|http_request|Go-http-client|^$`
|
|
||||||
|
|
||||||
// JSONResponse maps data as json
|
// JSONResponse maps data as json
|
||||||
type JSONResponse struct {
|
type JSONResponse struct {
|
||||||
IP string `json:"ip"`
|
IP string `json:"ip"`
|
||||||
@ -33,15 +30,17 @@ type JSONResponse struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func getRoot(ctx *gin.Context) {
|
func getRoot(ctx *gin.Context) {
|
||||||
reg := regexp.MustCompile(userAgentPattern)
|
switch ctx.NegotiateFormat(gin.MIMEPlain, gin.MIMEHTML, gin.MIMEJSON) {
|
||||||
if reg.Match([]byte(ctx.Request.UserAgent())) {
|
case gin.MIMEHTML:
|
||||||
ctx.String(http.StatusOK, ctx.ClientIP())
|
|
||||||
} else {
|
|
||||||
name := "home"
|
name := "home"
|
||||||
if setting.App.TemplatePath != "" {
|
if setting.App.TemplatePath != "" {
|
||||||
name = filepath.Base(setting.App.TemplatePath)
|
name = filepath.Base(setting.App.TemplatePath)
|
||||||
}
|
}
|
||||||
ctx.HTML(http.StatusOK, name, jsonOutput(ctx))
|
ctx.HTML(http.StatusOK, name, jsonOutput(ctx))
|
||||||
|
case gin.MIMEJSON:
|
||||||
|
getJSON(ctx)
|
||||||
|
default:
|
||||||
|
ctx.String(http.StatusOK, ctx.ClientIP()+"\n")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,31 +9,80 @@ import (
|
|||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestIP4RootFromCli(t *testing.T) {
|
func TestRootContentType(t *testing.T) {
|
||||||
uas := []string{
|
tests := []struct {
|
||||||
"",
|
name string
|
||||||
"curl",
|
accepted string
|
||||||
"wget",
|
expected string
|
||||||
"libwww-perl",
|
}{
|
||||||
"python",
|
{
|
||||||
"ansible-httpget",
|
name: "Accept wildcard",
|
||||||
"HTTPie",
|
accepted: "*/*",
|
||||||
"WindowsPowerShell",
|
expected: contentType.text,
|
||||||
"http_request",
|
},
|
||||||
"Go-http-client",
|
{
|
||||||
|
name: "Bogus accept",
|
||||||
|
accepted: "bogus/plain",
|
||||||
|
expected: contentType.text,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Accept plain text",
|
||||||
|
accepted: "text/plain",
|
||||||
|
expected: contentType.text,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Accept json",
|
||||||
|
accepted: "application/json",
|
||||||
|
expected: contentType.json,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
req, _ := http.NewRequest("GET", "/", nil)
|
||||||
|
req.Header.Set(trustedHeader, testIP.ipv4)
|
||||||
|
req.Header.Set("Accept", tt.accepted)
|
||||||
|
|
||||||
req, _ := http.NewRequest("GET", "/", nil)
|
w := httptest.NewRecorder()
|
||||||
req.Header.Set("X-Real-IP", testIP.ipv4)
|
app.ServeHTTP(w, req)
|
||||||
|
|
||||||
for _, ua := range uas {
|
assert.Equal(t, 200, w.Code)
|
||||||
req.Header.Set("User-Agent", ua)
|
assert.Equal(t, tt.expected, w.Header().Get("Content-Type"))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
w := httptest.NewRecorder()
|
func TestGetIP(t *testing.T) {
|
||||||
app.ServeHTTP(w, req)
|
expected := testIP.ipv4 + "\n"
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
accepted string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "No browser",
|
||||||
|
accepted: "*/*",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Bogus accept",
|
||||||
|
accepted: "bogus/plain",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Plain accept",
|
||||||
|
accepted: "text/plain",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
req, _ := http.NewRequest("GET", "/", nil)
|
||||||
|
req.Header.Set(trustedHeader, testIP.ipv4)
|
||||||
|
req.Header.Set("Accept", tt.accepted)
|
||||||
|
|
||||||
assert.Equal(t, 200, w.Code)
|
w := httptest.NewRecorder()
|
||||||
assert.Equal(t, testIP.ipv4, w.Body.String())
|
app.ServeHTTP(w, req)
|
||||||
|
|
||||||
|
assert.Equal(t, 200, w.Code)
|
||||||
|
assert.Equal(t, expected, w.Body.String())
|
||||||
|
assert.Equal(t, contentType.text, w.Header().Get("Content-Type"))
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -50,7 +99,7 @@ func TestHost(t *testing.T) {
|
|||||||
func TestClientPort(t *testing.T) {
|
func TestClientPort(t *testing.T) {
|
||||||
req, _ := http.NewRequest("GET", "/client-port", nil)
|
req, _ := http.NewRequest("GET", "/client-port", nil)
|
||||||
req.RemoteAddr = net.JoinHostPort(testIP.ipv4, "1000")
|
req.RemoteAddr = net.JoinHostPort(testIP.ipv4, "1000")
|
||||||
req.Header.Set("X-Real-IP", testIP.ipv4)
|
req.Header.Set(trustedHeader, testIP.ipv4)
|
||||||
|
|
||||||
w := httptest.NewRecorder()
|
w := httptest.NewRecorder()
|
||||||
app.ServeHTTP(w, req)
|
app.ServeHTTP(w, req)
|
||||||
@ -71,31 +120,44 @@ func TestNotFound(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestJSON(t *testing.T) {
|
func TestJSON(t *testing.T) {
|
||||||
expectedIPv4 := `{"client_port":"1000","ip":"81.2.69.192","ip_version":4,"country":"United Kingdom","country_code":"GB","city":"London","latitude":51.5142,"longitude":-0.0931,"postal_code":"","time_zone":"Europe/London","asn":0,"asn_organization":"","host":"test","headers":{"X-Real-Ip":["81.2.69.192"]}}`
|
type args struct {
|
||||||
expectedIPv6 := `{"asn":3352, "asn_organization":"TELEFONICA DE ESPANA", "city":"", "client_port":"1000", "country":"", "country_code":"", "headers":{"X-Real-Ip":["2a02:9000::1"]}, "host":"test", "ip":"2a02:9000::1", "ip_version":6, "latitude":0, "longitude":0, "postal_code":"", "time_zone":""}`
|
ip string
|
||||||
|
}
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
args args
|
||||||
|
expected string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "IPv4",
|
||||||
|
args: args{
|
||||||
|
ip: testIP.ipv4,
|
||||||
|
},
|
||||||
|
expected: jsonIPv4,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "IPv6",
|
||||||
|
args: args{
|
||||||
|
ip: testIP.ipv6,
|
||||||
|
},
|
||||||
|
expected: jsonIPv6,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
req, _ := http.NewRequest("GET", "/json", nil)
|
||||||
|
req.RemoteAddr = net.JoinHostPort(tt.args.ip, "1000")
|
||||||
|
req.Host = "test"
|
||||||
|
req.Header.Set(trustedHeader, tt.args.ip)
|
||||||
|
|
||||||
req, _ := http.NewRequest("GET", "/json", nil)
|
w := httptest.NewRecorder()
|
||||||
req.RemoteAddr = net.JoinHostPort(testIP.ipv4, "1000")
|
app.ServeHTTP(w, req)
|
||||||
req.Host = "test"
|
|
||||||
req.Header.Set("X-Real-IP", testIP.ipv4)
|
|
||||||
|
|
||||||
w := httptest.NewRecorder()
|
assert.Equal(t, 200, w.Code)
|
||||||
app.ServeHTTP(w, req)
|
assert.Equal(t, contentType.json, w.Header().Get("Content-Type"))
|
||||||
|
assert.JSONEq(t, tt.expected, w.Body.String())
|
||||||
assert.Equal(t, 200, w.Code)
|
})
|
||||||
assert.Equal(t, contentType.json, w.Header().Get("Content-Type"))
|
}
|
||||||
assert.JSONEq(t, expectedIPv4, w.Body.String())
|
|
||||||
|
|
||||||
req.RemoteAddr = net.JoinHostPort(testIP.ipv6, "1000")
|
|
||||||
req.Host = "test"
|
|
||||||
req.Header.Set("X-Real-IP", testIP.ipv6)
|
|
||||||
|
|
||||||
w = httptest.NewRecorder()
|
|
||||||
app.ServeHTTP(w, req)
|
|
||||||
|
|
||||||
assert.Equal(t, 200, w.Code)
|
|
||||||
assert.Equal(t, contentType.json, w.Header().Get("Content-Type"))
|
|
||||||
assert.JSONEq(t, expectedIPv6, w.Body.String())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestAll(t *testing.T) {
|
func TestAll(t *testing.T) {
|
||||||
@ -120,7 +182,7 @@ X-Real-Ip: 81.2.69.192
|
|||||||
req, _ := http.NewRequest("GET", "/all", nil)
|
req, _ := http.NewRequest("GET", "/all", nil)
|
||||||
req.RemoteAddr = net.JoinHostPort(testIP.ipv4, "1000")
|
req.RemoteAddr = net.JoinHostPort(testIP.ipv4, "1000")
|
||||||
req.Host = "test"
|
req.Host = "test"
|
||||||
req.Header.Set("X-Real-IP", testIP.ipv4)
|
req.Header.Set(trustedHeader, testIP.ipv4)
|
||||||
req.Header.Set("Header1", "one")
|
req.Header.Set("Header1", "one")
|
||||||
|
|
||||||
w := httptest.NewRecorder()
|
w := httptest.NewRecorder()
|
||||||
|
@ -16,6 +16,7 @@ type testIPs struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type contentTypes struct {
|
type contentTypes struct {
|
||||||
|
html string
|
||||||
text string
|
text string
|
||||||
json string
|
json string
|
||||||
}
|
}
|
||||||
@ -29,9 +30,12 @@ var (
|
|||||||
ipv6ASN: "2a02:a800::1",
|
ipv6ASN: "2a02:a800::1",
|
||||||
}
|
}
|
||||||
contentType = contentTypes{
|
contentType = contentTypes{
|
||||||
|
html: "content-type: text/html; charset=utf-8",
|
||||||
text: "text/plain; charset=utf-8",
|
text: "text/plain; charset=utf-8",
|
||||||
json: "application/json; charset=utf-8",
|
json: "application/json; charset=utf-8",
|
||||||
}
|
}
|
||||||
|
jsonIPv4 = `{"client_port":"1000","ip":"81.2.69.192","ip_version":4,"country":"United Kingdom","country_code":"GB","city":"London","latitude":51.5142,"longitude":-0.0931,"postal_code":"","time_zone":"Europe/London","asn":0,"asn_organization":"","host":"test","headers":{"X-Real-Ip":["81.2.69.192"]}}`
|
||||||
|
jsonIPv6 = `{"asn":3352, "asn_organization":"TELEFONICA DE ESPANA", "city":"", "client_port":"1000", "country":"", "country_code":"", "headers":{"X-Real-Ip":["2a02:9000::1"]}, "host":"test", "ip":"2a02:9000::1", "ip_version":6, "latitude":0, "longitude":0, "postal_code":"", "time_zone":""}`
|
||||||
)
|
)
|
||||||
|
|
||||||
const trustedHeader = "X-Real-IP"
|
const trustedHeader = "X-Real-IP"
|
||||||
|
Loading…
Reference in New Issue
Block a user