ipservice.go 1.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  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(http.MethodGet)
  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. // Responses:
  23. // 200: byteArrayResponse
  24. func getPublicIP(w http.ResponseWriter, r *http.Request) {
  25. r.Header.Set("Connection", "close")
  26. ip, err := parseIP(r)
  27. if err != nil {
  28. w.WriteHeader(400)
  29. switch {
  30. case ip != "":
  31. _, _ = w.Write([]byte("ip is invalid: " + ip))
  32. case ip == "":
  33. _, _ = w.Write([]byte("no ip found"))
  34. default:
  35. fmt.Println(err)
  36. }
  37. return
  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. }