ipservice.go 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  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. // swagger:route GET /api/getip ipservice getPublicIP
  14. //
  15. // Get the current public IP address
  16. //
  17. // Schemes: https
  18. //
  19. // Security:
  20. // oauth
  21. //
  22. func getPublicIP(w http.ResponseWriter, r *http.Request) {
  23. r.Header.Set("Connection", "close")
  24. ip, err := parseIP(r)
  25. if err != nil {
  26. w.WriteHeader(400)
  27. if ip != "" {
  28. w.Write([]byte("ip is invalid: " + ip))
  29. return
  30. } else {
  31. w.Write([]byte("no ip found"))
  32. return
  33. }
  34. } else {
  35. if err != nil {
  36. fmt.Println(err)
  37. }
  38. }
  39. w.WriteHeader(200)
  40. w.Write([]byte(ip))
  41. }
  42. func parseIP(r *http.Request) (string, error) {
  43. // Get Public IP from header
  44. ip := r.Header.Get("X-REAL-IP")
  45. ipnet := net.ParseIP(ip)
  46. if ipnet != nil && !ncutils.IpIsPrivate(ipnet) {
  47. return ip, nil
  48. }
  49. // If above fails, get Public IP from other header instead
  50. forwardips := r.Header.Get("X-FORWARDED-FOR")
  51. iplist := strings.Split(forwardips, ",")
  52. for _, ip := range iplist {
  53. ipnet := net.ParseIP(ip)
  54. if ipnet != nil && !ncutils.IpIsPrivate(ipnet) {
  55. return ip, nil
  56. }
  57. }
  58. // If above also fails, get Public IP from Remote Address of request
  59. ip, _, err := net.SplitHostPort(r.RemoteAddr)
  60. if err != nil {
  61. return "", err
  62. }
  63. ipnet = net.ParseIP(ip)
  64. if ipnet != nil {
  65. if ncutils.IpIsPrivate(ipnet) {
  66. return ip, fmt.Errorf("ip is a private address")
  67. }
  68. return ip, nil
  69. }
  70. return "", fmt.Errorf("no ip found")
  71. }