2
0

ipservice.go 1.7 KB

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