獲取真實IP地址——代理背后的終端ip地址


再做個筆記吧,系統操作日志、登錄日志之類的記錄數據通常都會記錄IP信息,但是如何獲取用戶真實IP而不是代理服務器的呢?考慮到用戶如果有2層,3層的代理,HTTP_X_FORWARDED_FOR 的值應該是:“本機真實IP,1層代理IP,2層代理IP,.....” ,而實際應用中,使用多層透明代理的情況可能比較少,所以這種用戶並不多。目前更多的網站使用了代理加速方式,使用Squid做代理,利用多台服務器分流。Squid本身類似透明代理,會發送“HTTP_X_FORWARDED_FOR” ,HTTP_X_FORWARDED_FOR 中包括客戶的IP地址,如果此時客戶已經使用了一層透明代理,那么程序取的 “HTTP_X_FORWARDED_FOR” 就包括兩個IP地址。所以獲取真實IP地址的方式,應該判斷“HTTP_X_FORWARDED_FOR”中是否有“,”逗號,或者長度是否超過15字節(xxx.xxx.xxx.xxx),多個則取第一個非內網IP。

using System;
using System.Web;
using System.Text.RegularExpressions;

namespace Core
{
    public class IPHelper
    {
        /// <summary> 
        /// 取得客戶端真實IP。如果有代理則取第一個非內網地址 
        /// </summary> 
        public static string IPAddress
        {
            get
            {
                string result = String.Empty;

                result = HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"];
                if (result != null && result != String.Empty)
                {
                    //可能有代理 
                    if (result.IndexOf(".") == -1)     //沒有“.”肯定是非IPv4格式 
                        result = null;
                    else
                    {
                        if (result.IndexOf(",") != -1)
                        {
                            //有“,”,估計多個代理。取第一個不是內網的IP。 
                            result = result.Replace(" ", "").Replace("'", "");
                            string[] temparyip = result.Split(",;".ToCharArray());
                            for (int i = 0; i < temparyip.Length; i++)
                            {
                                if (IsIPAddress(temparyip[i])
                                    && temparyip[i].Substring(0, 3) != "10."
                                    && temparyip[i].Substring(0, 7) != "192.168"
                                    && temparyip[i].Substring(0, 7) != "172.16.")
                                {
                                    return temparyip[i];     //找到不是內網的地址 
                                }
                            }
                        }
                        else if (IsIPAddress(result)) //代理即是IP格式 
                            return result;
                        else
                            result = null;     //代理中的內容 非IP,取IP 
                    }

                }

                string IpAddress =
                    (HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"] != null &&
                     HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"] != String.Empty)
                        ? HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"]
                        : HttpContext.Current.Request.ServerVariables["REMOTE_ADDR"];

                if (null == result || result == String.Empty)
                    result = HttpContext.Current.Request.ServerVariables["REMOTE_ADDR"];

                if (result == null || result == String.Empty)
                    result = HttpContext.Current.Request.UserHostAddress;

                return result;
            }
        }

        /// <summary>
        /// 判斷是否是IP地址格式
        /// </summary>
        /// <param name="str1">待判斷的IP地址</param>
        /// <returns>true or false</returns>
        public static bool IsIPAddress(string str1)
        {
            if (str1 == null || str1 == String.Empty || str1.Length < 7 || str1.Length > 15) return false;

            Regex regex = new Regex(@"^\d{1,3}[\.]\d{1,3}[\.]\d{1,3}[\.]\d{1,3}$", RegexOptions.IgnoreCase);
            return regex.IsMatch(str1);
        }
    }
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM