ipservice.go 1.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869
  1. package controller
  2. import (
  3. "fmt"
  4. "net"
  5. "net/http"
  6. "strings"
  7. "github.com/gorilla/mux"
  8. "github.com/gravitl/netmaker/netclient/ncutils"
  9. )
  10. func ipHandlers(r *mux.Router) {
  11. r.HandleFunc("/api/getip", http.HandlerFunc(getPublicIP)).Methods("GET")
  12. }
  13. func getPublicIP(w http.ResponseWriter, r *http.Request) {
  14. r.Header.Set("Connection", "close")
  15. ip, err := parseIP(r)
  16. if err != nil {
  17. w.WriteHeader(400)
  18. if ip != "" {
  19. w.Write([]byte("ip is invalid: " + ip))
  20. return
  21. } else {
  22. w.Write([]byte("no ip found"))
  23. return
  24. }
  25. } else {
  26. if err != nil {
  27. fmt.Println(err)
  28. }
  29. }
  30. w.WriteHeader(200)
  31. w.Write([]byte(ip))
  32. }
  33. func parseIP(r *http.Request) (string, error) {
  34. // Get Public IP from header
  35. ip := r.Header.Get("X-REAL-IP")
  36. ipnet := net.ParseIP(ip)
  37. if ipnet != nil && !ncutils.IpIsPrivate(ipnet) {
  38. return ip, nil
  39. }
  40. // If above fails, get Public IP from other header instead
  41. forwardips := r.Header.Get("X-FORWARDED-FOR")
  42. iplist := strings.Split(forwardips, ",")
  43. for _, ip := range iplist {
  44. ipnet := net.ParseIP(ip)
  45. if ipnet != nil && !ncutils.IpIsPrivate(ipnet) {
  46. return ip, nil
  47. }
  48. }
  49. // If above also fails, get Public IP from Remote Address of request
  50. ip, _, err := net.SplitHostPort(r.RemoteAddr)
  51. if err != nil {
  52. return "", err
  53. }
  54. ipnet = net.ParseIP(ip)
  55. if ipnet != nil {
  56. if ncutils.IpIsPrivate(ipnet) {
  57. return ip, fmt.Errorf("ip is a private address")
  58. }
  59. return ip, nil
  60. }
  61. return "", fmt.Errorf("no ip found")
  62. }