mirror of
https://github.com/mpolden/echoip.git
synced 2025-01-27 18:35:04 +01:00
parent
04383146ed
commit
fa32c7ff66
41
api/api.go
41
api/api.go
@ -29,20 +29,32 @@ const (
|
|||||||
var cliUserAgentExp = regexp.MustCompile("^(?i)(curl|wget|fetch\\slibfetch)\\/.*$")
|
var cliUserAgentExp = regexp.MustCompile("^(?i)(curl|wget|fetch\\slibfetch)\\/.*$")
|
||||||
|
|
||||||
type API struct {
|
type API struct {
|
||||||
db *geoip2.Reader
|
|
||||||
CORS bool
|
CORS bool
|
||||||
ReverseLookup bool
|
ReverseLookup bool
|
||||||
Template string
|
Template string
|
||||||
|
lookupAddr func(string) ([]string, error)
|
||||||
|
lookupCountry func(net.IP) (string, error)
|
||||||
|
ipFromRequest func(*http.Request) (net.IP, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
func New() *API { return &API{} }
|
func New() *API {
|
||||||
|
return &API{
|
||||||
|
lookupAddr: net.LookupAddr,
|
||||||
|
lookupCountry: func(ip net.IP) (string, error) { return "", nil },
|
||||||
|
ipFromRequest: ipFromRequest,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func NewWithGeoIP(filepath string) (*API, error) {
|
func NewWithGeoIP(filepath string) (*API, error) {
|
||||||
db, err := geoip2.Open(filepath)
|
db, err := geoip2.Open(filepath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return &API{db: db}, nil
|
api := New()
|
||||||
|
api.lookupCountry = func(ip net.IP) (string, error) {
|
||||||
|
return lookupCountry(db, ip)
|
||||||
|
}
|
||||||
|
return api, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type Cmd struct {
|
type Cmd struct {
|
||||||
@ -100,11 +112,11 @@ func headerPairFromRequest(r *http.Request) (string, string, error) {
|
|||||||
return header, value, nil
|
return header, value, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *API) lookupCountry(ip net.IP) (string, error) {
|
func lookupCountry(db *geoip2.Reader, ip net.IP) (string, error) {
|
||||||
if a.db == nil {
|
if db == nil {
|
||||||
return "", nil
|
return "", nil
|
||||||
}
|
}
|
||||||
record, err := a.db.Country(ip)
|
record, err := db.Country(ip)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
@ -187,25 +199,24 @@ func cliMatcher(r *http.Request, rm *mux.RouteMatch) bool {
|
|||||||
|
|
||||||
func (a *API) requestFilter(next http.Handler) http.Handler {
|
func (a *API) requestFilter(next http.Handler) http.Handler {
|
||||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
ip, err := ipFromRequest(r)
|
ip, err := a.ipFromRequest(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.Header.Set(IP_HEADER, err.Error())
|
log.Print(err)
|
||||||
|
r.Header.Set(IP_HEADER, "")
|
||||||
} else {
|
} else {
|
||||||
r.Header.Set(IP_HEADER, ip.String())
|
r.Header.Set(IP_HEADER, ip.String())
|
||||||
country, err := a.lookupCountry(ip)
|
country, err := a.lookupCountry(ip)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.Header.Set(COUNTRY_HEADER, err.Error())
|
log.Print(err)
|
||||||
} else {
|
|
||||||
r.Header.Set(COUNTRY_HEADER, country)
|
|
||||||
}
|
}
|
||||||
|
r.Header.Set(COUNTRY_HEADER, country)
|
||||||
}
|
}
|
||||||
if a.ReverseLookup {
|
if a.ReverseLookup {
|
||||||
hostname, err := net.LookupAddr(ip.String())
|
hostname, err := a.lookupAddr(ip.String())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.Header.Set(HOSTNAME_HEADER, err.Error())
|
log.Print(err)
|
||||||
} else {
|
|
||||||
r.Header.Set(HOSTNAME_HEADER, strings.Join(hostname, ", "))
|
|
||||||
}
|
}
|
||||||
|
r.Header.Set(HOSTNAME_HEADER, strings.Join(hostname, ", "))
|
||||||
}
|
}
|
||||||
if a.CORS {
|
if a.CORS {
|
||||||
w.Header().Set("Access-Control-Allow-Methods", "GET")
|
w.Header().Set("Access-Control-Allow-Methods", "GET")
|
||||||
|
@ -9,9 +9,25 @@ import (
|
|||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
"net/url"
|
"net/url"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func newTestAPI() *API {
|
||||||
|
return &API{
|
||||||
|
lookupAddr: func(string) ([]string, error) {
|
||||||
|
return []string{"localhost"}, nil
|
||||||
|
},
|
||||||
|
lookupCountry: func(ip net.IP) (string, error) {
|
||||||
|
return "Elbonia", nil
|
||||||
|
},
|
||||||
|
ipFromRequest: func(*http.Request) (net.IP, error) {
|
||||||
|
return net.ParseIP("127.0.0.1"), nil
|
||||||
|
},
|
||||||
|
ReverseLookup: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func httpGet(url string, json bool, userAgent string) (string, int, error) {
|
func httpGet(url string, json bool, userAgent string) (string, int, error) {
|
||||||
r, err := http.NewRequest("GET", url, nil)
|
r, err := http.NewRequest("GET", url, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -38,9 +54,10 @@ func TestGetIP(t *testing.T) {
|
|||||||
toJSON := func(k string, v string) string {
|
toJSON := func(k string, v string) string {
|
||||||
return fmt.Sprintf("{\n \"%s\": \"%s\"\n}", k, v)
|
return fmt.Sprintf("{\n \"%s\": \"%s\"\n}", k, v)
|
||||||
}
|
}
|
||||||
s := httptest.NewServer(New().Handlers())
|
s := httptest.NewServer(newTestAPI().Handlers())
|
||||||
jsonAll := "{\n \"Accept-Encoding\": [\n \"gzip\"\n ]," +
|
jsonAll := "{\n \"Accept-Encoding\": [\n \"gzip\"\n ]," +
|
||||||
"\n \"X-Ifconfig-Country\": [\n \"\"\n ]," +
|
"\n \"X-Ifconfig-Country\": [\n \"Elbonia\"\n ]," +
|
||||||
|
"\n \"X-Ifconfig-Hostname\": [\n \"localhost\"\n ]," +
|
||||||
"\n \"X-Ifconfig-Ip\": [\n \"127.0.0.1\"\n ]\n}"
|
"\n \"X-Ifconfig-Ip\": [\n \"127.0.0.1\"\n ]\n}"
|
||||||
|
|
||||||
var tests = []struct {
|
var tests = []struct {
|
||||||
@ -74,6 +91,21 @@ func TestGetIP(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestGetIPWithoutReverse(t *testing.T) {
|
||||||
|
log.SetOutput(ioutil.Discard)
|
||||||
|
api := newTestAPI()
|
||||||
|
api.ReverseLookup = false
|
||||||
|
s := httptest.NewServer(api.Handlers())
|
||||||
|
|
||||||
|
out, _, err := httpGet(s.URL, false, "curl/7.26.0")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if header := "X-Ifconfig-Hostname"; strings.Contains(out, header) {
|
||||||
|
t.Errorf("Expected response to not contain %q", header)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestIPFromRequest(t *testing.T) {
|
func TestIPFromRequest(t *testing.T) {
|
||||||
var tests = []struct {
|
var tests = []struct {
|
||||||
in *http.Request
|
in *http.Request
|
||||||
|
Loading…
x
Reference in New Issue
Block a user