Go http real ip header parser module
A forwarders such as a reverse proxy or Cloudflare find the real IP address from the requests made to the http server behind it. Local IP addresses and CloudFlare ip addresses are defined by default within the module. It is possible to define more forwarder IP addresses.
In Turkey, it is obligatory to keep the port information of IP addresses shared with cgnat by the law no 5651. For this reason, dst port information is also given along with the IP addresses. If the IP address is behind a proxy, the dst port information is returned as -1 — the port on r.RemoteAddr belongs to the proxy's own connection, not the real client, so exposing it would be misleading. Some proxies forward the real client port in a header (Cloudflare Cf-Connecting-Port, Heimwall True-Client-Port); when a request comes from a trusted forwarder, that header is read and validated (must be a 1-65535 TCP port), otherwise the port stays -1. Additional port headers can be registered with AddPortHeaders.
go get -u github.com/netinternet/remoteaddr
// remoteaddr.Parse().IP(*http.Request) return to string IPv4 or IPv6 addressRun a simple web server and get the real IP address to string format
package main
import (
"fmt"
"log"
"net/http"
"github.com/netinternet/remoteaddr"
)
func root(w http.ResponseWriter, r *http.Request) {
ip, port := remoteaddr.Parse().IP(r)
fmt.Fprintf(w, "Your IP address is "+ip+" and dst port "+port)
}
func main() {
http.HandleFunc("/", root)
log.Fatal(http.ListenAndServe(":8081", nil))
}AddForwarders([]string{"8.8.8.0/24"}) = Add a new multiple forwarder prefixes
package main
import (
"fmt"
"log"
"net/http"
"github.com/netinternet/remoteaddr"
)
func root(w http.ResponseWriter, r *http.Request) {
ip, port := remoteaddr.Parse().AddForwarders([]string{"8.8.8.0/24"}).IP(r)
fmt.Fprintf(w, "Your IP address is "+ip+" and dst port "+port)
}
func main() {
http.HandleFunc("/", root)
log.Fatal(http.ListenAndServe(":8081", nil))
}AddPortHeaders([]string{"X-Real-Port"}) = Add a new multiple real client port headers (defaults already include Cf-Connecting-Port and True-Client-Port)
func root(w http.ResponseWriter, r *http.Request) {
ip, port := remoteaddr.Parse().AddPortHeaders([]string{"X-Real-Port"}).IP(r)
fmt.Fprintf(w, "Your IP address is "+ip+" and dst port "+port)
}AddHeaders([]string{"True-Client-IP"}) = Add a new multiple real ip headers
package main
import (
"fmt"
"log"
"net/http"
"github.com/netinternet/remoteaddr"
)
func root(w http.ResponseWriter, r *http.Request) {
ip, port := remoteaddr.Parse().AddHeaders([]string{"True-Client-IP"}).IP(r)
fmt.Fprintf(w, "Your IP address is "+ip+" and dst port "+port)
}
func main() {
http.HandleFunc("/", root)
log.Fatal(http.ListenAndServe(":8081", nil))
}